javascript
SpringBoot webmvc项目导出war包并在外部tomcat运行产生的诸多问题以及解决方案
背景:
有需求要將原來的Spring(3.2.6) + Springmvc + Hibernate項(xiàng)目重構(gòu)為Springboot(1.5.2)項(xiàng)目
描述:
記錄重構(gòu)過程,以及期間遇到的種種問題和對(duì)應(yīng)的解決方案
環(huán)境:
原項(xiàng)目: win10 + eclipse + jdk1.8 + mysql5.7
新項(xiàng)目: win10 + IDEA + jdk1.8?+ mysql5.7 + Maven
過程:
第一步: ?新建Maven項(xiàng)目
IDEA: project > New > Module > Maven (選擇 maven-archetype-quickstart 快速創(chuàng)建一個(gè)maven項(xiàng)目, 如下圖)
點(diǎn)擊Next, 自己想一個(gè)項(xiàng)目的 groupid(一般為項(xiàng)目域名的倒寫) 和 artifactid(項(xiàng)目名) 并填好(如下圖)
點(diǎn)擊Next, 確認(rèn)創(chuàng)建信息
點(diǎn)擊Next, 選擇項(xiàng)目創(chuàng)建文件夾地址
點(diǎn)擊確認(rèn)自動(dòng)創(chuàng)建項(xiàng)目
項(xiàng)目創(chuàng)建就完成了
如果發(fā)現(xiàn)創(chuàng)建maven項(xiàng)目十分緩慢, 很可能是由于訪問maven官方中央倉庫網(wǎng)速太差導(dǎo)致的,建議可以修改Maven的settings.xml文件
將默認(rèn)的倉庫地址改為國內(nèi)阿里云的地址(http://maven.aliyun.com/nexus/content/groups/public/),(如下圖)
?
第二步: 配置pom.xml
不多說,上代碼,如果對(duì)其中某些節(jié)點(diǎn)含義不清楚, 可以參考此博文:?https://www.cnblogs.com/hafiz/p/5360195.html
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><groupId>com.yjy.test</groupId><version>1.0-SNAPSHOT</version><artifactId>yjyboot-${project.version}</artifactId><name>yjyboot</name><packaging>war</packaging><properties><java.version>1.8</java.version><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding><log4j2.level>debug</log4j2.level><log4j2.root.path>/logs/${project.name}</log4j2.root.path><log4j2.error.path>/logs/${project.name}-error</log4j2.error.path><log4j2.package.path>/logs/${project.name}-kk</log4j2.package.path></properties><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>1.5.2.RELEASE</version></parent><dependencies><dependency><groupId>junit</groupId><artifactId>junit</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><exclusions><exclusion><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-logging</artifactId></exclusion></exclusions></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-devtools</artifactId><optional>true</optional></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-configuration-processor</artifactId><optional>true</optional></dependency><!--https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-log4j2--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-log4j2</artifactId></dependency><!-- https://mvnrepository.com/artifact/org.apache.logging.log4j/log4j-web --><!-- 如果沒有此 log4j-web 導(dǎo)出的war將不能打印日志到文件!!! --><dependency><groupId>org.apache.logging.log4j</groupId><artifactId>log4j-web</artifactId><version>2.7</version></dependency><!-- freemarker --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-freemarker</artifactId></dependency><!-- 下面兩個(gè)引入為了操作數(shù)據(jù)庫 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jpa</artifactId></dependency><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId></dependency><!-- Json包 --><dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.16</version></dependency><!-- 為了監(jiān)控?cái)?shù)據(jù)庫 --><dependency><groupId>com.alibaba</groupId><artifactId>druid</artifactId><version>1.0.25</version></dependency><!-- commons --><dependency><groupId>org.apache.commons</groupId><artifactId>commons-lang3</artifactId><version>3.7</version></dependency><!-- httpclient --><dependency><groupId>org.apache.httpcomponents</groupId><artifactId>httpmime</artifactId><version>4.5.3</version></dependency><!-- https://mvnrepository.com/artifact/commons-httpclient/commons-httpclient --><dependency><groupId>commons-httpclient</groupId><artifactId>commons-httpclient</artifactId><version>3.1</version></dependency><dependency><groupId>commons-beanutils</groupId><artifactId>commons-beanutils</artifactId></dependency><dependency><groupId>commons-io</groupId><artifactId>commons-io</artifactId><version>2.5</version></dependency><!-- 兼容log4j --><dependency><groupId>org.apache.logging.log4j</groupId><artifactId>log4j-1.2-api</artifactId><version>2.8.2</version></dependency><!-- Redis --><dependency><groupId>redis.clients</groupId><artifactId>jedis</artifactId></dependency><!-- https://mvnrepository.com/artifact/nl.bitwalker/UserAgentUtils --><dependency><groupId>nl.bitwalker</groupId><artifactId>UserAgentUtils</artifactId><version>1.2.4</version></dependency><!--打war包時(shí)加入此項(xiàng) 告訴spring-boot tomcat相關(guān)jar包用外部的 不要打進(jìn)去IDEA運(yùn)行時(shí)需要將此依賴注釋掉, 否則會(huì)無法運(yùn)行--><!--<dependency>--><!--<groupId>org.springframework.boot</groupId>--><!--<artifactId>spring-boot-starter-tomcat</artifactId>--><!--<scope>provided</scope>--><!--</dependency>--></dependencies><build><finalName>${project.name}</finalName><directory>target</directory><sourceDirectory>src/main/java</sourceDirectory><testSourceDirectory>src/test/java</testSourceDirectory><outputDirectory>target</outputDirectory><resources><resource><directory>src/main/resources</directory><filtering>true</filtering><includes><include>**/*</include></includes></resource><!-- 將自定義的Servlet(extends DispatcherServlet)默認(rèn)xml配置文件打包至WEB-INF下否則外部tomcat無法處理此Servlet --><resource><directory>src/main/extraConfig</directory><targetPath>${build.finalName}/WEB-INF/</targetPath></resource></resources><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><executions><execution><goals><goal>build-info</goal></goals></execution></executions></plugin><!--spring-boot為了保護(hù)application.yml和application.properties,修改了默認(rèn)的占位符${...}為@...@--><!--為了spring boot的yml和properties文件能夠使用maven變量替換,使用${}占位符--><plugin><artifactId>maven-resources-plugin</artifactId><configuration><encoding>utf-8</encoding><useDefaultDelimiters>true</useDefaultDelimiters></configuration></plugin></plugins></build></project> pom.xml
第三步: 創(chuàng)建配置文件, 特別注意: ?log4j2.xml 只能放在resources目錄下, 否則導(dǎo)出的war包,配置的log4j2將不起作用!!!!
spring:profiles:active: dev # 激活的配置文件include: freemarker,mysql,redis,interceptor # 加載其他配置文件mvc:favicon:enabled: truedebug: true # 是否啟用debugserver:servlet-path: /common # 所有接口請(qǐng)求都交由自定義的Servlet處理了, 所以默認(rèn)的servlet只用于處理靜態(tài)資源 application.yml spring:freemarker:enabled: true # 是否啟用freemarkercache: false # 是否啟用緩存prefix: # 模板文件前綴suffix: .ftl # 模板文件后綴charset: UTF-8 # 模板文件編碼template-loader-path: classpath:templates/ # 模板文件目錄check-template-location: true # 是否檢查模板目錄是否存在content-type: text/html # 模板類型request-context-attribute: req # RequestContext 引用settings: # 更多配置number_format: '0.##' #數(shù)字格式化, 保留兩位小數(shù)allow-request-override: false # 是否允許 request 屬性覆蓋 controller 屬性allow-session-override: false # 是否允許 session 屬性覆蓋 controller 屬性expose-request-attributes: false # 設(shè)置在與模板合并之前,是否應(yīng)該將所有HttpRequest屬性添加到模型中。expose-session-attributes: false # 設(shè)置在與模板合并之前,是否應(yīng)該將所有HttpSession屬性添加到模型中。expose-spring-macro-helpers: true # 設(shè)置是否公開一個(gè)請(qǐng)求上下文,以供Spring的宏庫使用,名稱為“springMacroRequestContext”prefer-file-system-access: true # 更喜歡文件系統(tǒng)訪問模板加載。文件系統(tǒng)訪問支持對(duì)模板更改進(jìn)行熱檢測(cè)。 # view-names: # whitelist of view names that can be resolved application-freemarker.yml front:login:excludeUrls: # 這里配置的前臺(tái)登入驗(yàn)證的白名單- /hello.sv- /hello2.sv- /index.jtk- /autho.jtk- /code.jtk- /checkLogin.jtk- /checkUser.jtk- /test.jtk- /wxPay/notify.jtk- /api/list.jtk- /api/deposit.jtk- /config/wechat.jtkback:login:excludeUrls: # 這里配置的后臺(tái)登入驗(yàn)證的白名單- /login.do- /logout.do- /game/scores.do- /game/dayScore.do application-interceptor.yml spring:datasource:# 數(shù)據(jù)庫訪問配置# 主數(shù)據(jù)源,默認(rèn)的type: com.alibaba.druid.pool.DruidDataSourcedbUrl: jdbc:mysql://localhost:3306/hotpot?useUnicode=true&characterEncoding=utf-8&useSSL=falseusername: yjypassword: yyyyyydriverClassName: com.mysql.jdbc.Driver# 下面為連接池的補(bǔ)充設(shè)置,應(yīng)用到上面所有數(shù)據(jù)源中# 初始化大小,最小,最大initialSize: 5minIdle: 5maxActive: 20maxWait: 60000 # 配置獲取連接等待超時(shí)的時(shí)間timeBetweenEvictionRunsMillis: 60000 # 配置間隔多久才進(jìn)行一次檢測(cè),檢測(cè)需要關(guān)閉的空閑連接,單位是毫秒minEvictableIdleTimeMillis: 300000 # 配置一個(gè)連接在池中最小生存的時(shí)間,單位是毫秒validationQuery: SELECT 1 FROM DUALtestWhileIdle: truetestOnBorrow: falsetestOnReturn: falsepoolPreparedStatements: true # 打開PSCache,并且指定每個(gè)連接上PSCache的大小maxPoolPreparedStatementPerConnectionSize: 20filters: stat,wall,log4j # 配置監(jiān)控統(tǒng)計(jì)攔截的filters,去掉后監(jiān)控界面sql無法統(tǒng)計(jì),'wall'用于防火墻connectionProperties: druid.stat.mergeSql=true;druid.stat.slowSqlMillis=5000 # 通過connectProperties屬性來打開mergeSql功能;慢SQL記錄useGlobalDataSourceStat: true # 合并多個(gè)DruidDataSource的監(jiān)控?cái)?shù)據(jù)#JPA Configuration:jpa:database: MYSQLshow-sql: true # Show or not log for each sql querygenerate-ddl: true # Hibernate ddl auto (create, create-drop, update)hibernate:ddl-auto: updatenaming:strategy: org.hibernate.cfg.ImprovedNamingStrategyproperties:hibernate:dialect: org.hibernate.dialect.MySQL5Dialect application-mysql.yml <?xml version="1.0" encoding="UTF-8"?> <!--日志級(jí)別以及優(yōu)先級(jí)排序: OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL --> <!--Configuration后面的status,這個(gè)用于設(shè)置log4j2自身內(nèi)部的信息輸出,可以不設(shè)置,當(dāng)設(shè)置成trace時(shí),你會(huì)看到log4j2內(nèi)部各種詳細(xì)輸出--> <!--monitorInterval:Log4j能夠自動(dòng)檢測(cè)修改配置 文件和重新配置本身,設(shè)置間隔秒數(shù)--> <configuration status="INFO" monitorInterval="30"><!--先定義所有的appender--><appenders><!--這個(gè)輸出控制臺(tái)的配置--><console name="Console" target="SYSTEM_OUT"><!--輸出日志的格式--><PatternLayout pattern="[%d{HH:mm:ss:SSS}] [%p] - %l - %m%n"/><!--控制臺(tái)只輸出level及以上級(jí)別的信息(onMatch),其他的直接拒絕(onMismatch)--><ThresholdFilter level="INFO" onMatch="ACCEPT" onMismatch="DENY"/></console><!--文件會(huì)打印出所有信息,這個(gè)log每次運(yùn)行程序會(huì)自動(dòng)清空,由append屬性決定,這個(gè)也挺有用的,適合臨時(shí)測(cè)試用--><File name="CurrentLog" fileName="logs/current.log" append="false"><PatternLayout pattern="%d{HH:mm:ss.SSS} %-5level %class{36} %L %M - %msg%xEx%n"/></File><!-- 這個(gè)會(huì)打印出所有的info及以下級(jí)別的信息,每次大小超過size,則這size大小的日志會(huì)自動(dòng)存入按年份-月份建立的文件夾下面并進(jìn)行壓縮,作為存檔--><RollingFile name="RollingFileInfo" fileName="F:/logs/info.log"filePattern="F:/logs/$${date:yyyy-MM}/info-%d{yyyy-MM-dd}-%i.log"><!--控制臺(tái)只輸出level及以上級(jí)別的信息(onMatch),其他的直接拒絕(onMismatch)--><ThresholdFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/><PatternLayout pattern="[%d{HH:mm:ss:SSS}] [%p] - %l - %m%n"/><Policies><TimeBasedTriggeringPolicy/><SizeBasedTriggeringPolicy size="50MB"/></Policies></RollingFile><RollingFile name="RollingFileWarn" fileName="F:/logs/warn.log"filePattern="F:/logs/$${date:yyyy-MM}/warn-%d{yyyy-MM-dd}-%i.log"><ThresholdFilter level="warn" onMatch="ACCEPT" onMismatch="DENY"/><PatternLayout pattern="[%d{HH:mm:ss:SSS}] [%p] - %l - %m%n"/><Policies><TimeBasedTriggeringPolicy/><SizeBasedTriggeringPolicy size="30MB"/></Policies><!-- DefaultRolloverStrategy屬性如不設(shè)置,則默認(rèn)為最多同一文件夾下7個(gè)文件,這里設(shè)置了20 --><DefaultRolloverStrategy max="20"/></RollingFile><RollingFile name="RollingFileError" fileName="F:/logs/error.log"filePattern="F:/logs/$${date:yyyy-MM}/error-%d{yyyy-MM-dd}-%i.log"><ThresholdFilter level="error" onMatch="ACCEPT" onMismatch="DENY"/><PatternLayout pattern="[%d{HH:mm:ss:SSS}] [%p] - %l - %m%n"/><Policies><TimeBasedTriggeringPolicy/><SizeBasedTriggeringPolicy size="20MB"/></Policies></RollingFile></appenders><!--然后定義logger,只有定義了logger并引入的appender,appender才會(huì)生效--><loggers><!--過濾掉spring和mybatis的一些無用的DEBUG信息--><logger name="org.springframework" level="INFO"/><logger name="org.springframework.boot.autoconfigure.logging" level="INFO"/><logger name="org.springframework.boot.logging" level="INFO"/><logger name="org.mybatis" level="INFO"/><logger name="org.hibernate" level="INFO"/><logger name="druid.sql" level="INFO"/><root level="info"><appender-ref ref="Console"/><appender-ref ref="CurrentLog"/><appender-ref ref="RollingFileInfo"/><appender-ref ref="RollingFileWarn"/><appender-ref ref="RollingFileError"/></root></loggers> </configuration> log4j2.xml server:port: 8082 # 嵌入server的運(yùn)行端口context-path: /yjyboot # 配置項(xiàng)目運(yùn)行地址 spring:devtools:restart:exclude: classpath:common/**,classpath:templates/** application-dev.yml server:port: 8080context-path: /yjyboot # 導(dǎo)出war包存放在tomcat后會(huì)有一個(gè)項(xiàng)目運(yùn)行地址, 這里配置可以模擬項(xiàng)目地址, 達(dá)到IDEA運(yùn)行與tomcat運(yùn)行地址相同 application-pro.yml
?
第四步: 主類(一般放在根包中)
package com.yjy.test;import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.domain.EntityScan; import org.springframework.boot.builder.SpringApplicationBuilder; import org.springframework.boot.web.servlet.ServletComponentScan; import org.springframework.boot.web.support.SpringBootServletInitializer; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.PropertySource;/*** SpringBoot 啟動(dòng)入口** @Author yjy* @Date 2018-04-17 12:43*/ //@SpringBootApplication = (@Configuration, @EnableAutoConfiguration, @ComponentScan) @Configuration @EnableAutoConfiguration @ComponentScan @EntityScan("com.yjy.test.game.entity") // 掃描實(shí)體類 @ServletComponentScan(basePackages = "com.yjy.test") // 掃描自定義Servlet @PropertySource(value = { // 導(dǎo)入配置"classpath:/config/application.yml", }) public class Application extends SpringBootServletInitializer {// IDEA運(yùn)行時(shí) 運(yùn)行此函數(shù)public static void main(String[] args) {SpringApplication.run(Application.class, args);}// 導(dǎo)出war在外部tomcat使用時(shí), 不能使用main函數(shù)運(yùn)行, 需要配置此項(xiàng) @Overrideprotected SpringApplicationBuilder configure(SpringApplicationBuilder application) {return application.sources(Application.class);}}
第五步: 添加配置類(按自己的需要添加, 無特別說明的情況下, 配置類可以存在任意包內(nèi), 只需滿足包級(jí)別不高于Application.java所在的包就可以
當(dāng)然也可以通過配置掃描包注解來自定義, 默認(rèn)掃描主類所在包以下的所有包)
1: 全局跨域配置類(放在與Application.java同目錄下)
package com.yjy.test;import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.cors.CorsConfiguration; import org.springframework.web.cors.UrlBasedCorsConfigurationSource; import org.springframework.web.filter.CorsFilter;/*** 跨域配置** @Author yjy* @Date 2018-04-26 15:55*/ @Configuration public class CorsConfig {private CorsConfiguration buildConfig() {CorsConfiguration corsConfiguration = new CorsConfiguration();corsConfiguration.addAllowedHeader("*");corsConfiguration.addAllowedMethod("*");corsConfiguration.addAllowedOrigin("*");corsConfiguration.setAllowCredentials(true);return corsConfiguration;}@Beanpublic CorsFilter corsFilter() {UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();source.registerCorsConfiguration("/**", buildConfig());return new CorsFilter(source);}} 跨域配置?
2: Date參數(shù)的格式化( 請(qǐng)求中符合格式的字符串參數(shù)可以使用Date類型接收參數(shù), 比如請(qǐng)求參數(shù) ?addTime=20180101, Controller層可以使用 func(Date addTime); 接收, 否則會(huì)報(bào)400錯(cuò)誤)
package com.yjy.test;import org.apache.commons.lang3.StringUtils; import org.springframework.core.convert.converter.Converter; import org.springframework.stereotype.Component;import java.text.ParseException; import java.text.SimpleDateFormat;/*** Date參數(shù)格式化** 嘗試格式化java.util.Date類型的參數(shù)** @Author yjy* @Date 2018-04-27 9:58*/ @Component public class DateConverterConfig implements Converter<String, java.util.Date> {private static final String[] formats = new String[] {"yyyy-MM-dd", // 0"yyyy-MM", // 1"yyyy-MM-dd HH:mm:ss", // 2"yyyy-MM-dd HH:mm", // 3 };/*** 這里將參數(shù)格式化成 java.sql.Date 為了方便后面用來拼接sql* @param param 日期格式的字符串* @return java.sql.Date*/@Overridepublic java.sql.Date convert(String param) {if (StringUtils.isBlank(param)) {return null;}param = param.trim();if (param.matches("^\\d{4}-\\d{1,2}-\\d{1,2}$")) {return parseDate(param, formats[0]);}if (param.matches("^\\d{4}-\\d{1,2}$")) {return parseDate(param, formats[1]);}if (param.matches("^\\d{4}-\\d{1,2}-\\d{1,2} \\d{1,2}:\\d{1,2}:\\d{1,2}$")) {return parseDate(param, formats[2]);}if (param.matches("^\\d{4}-\\d{1,2}-\\d{1,2} \\d{1,2}:\\d{1,2}$")) {return parseDate(param, formats[3]);}throw new IllegalArgumentException("Invalid date param '" + param + "'");}/*** 格式化日期* @param dateStr 日期字符串* @param format 格式* @return 日期*/private java.sql.Date parseDate(String dateStr, String format) {java.sql.Date date = null;try {SimpleDateFormat simpleDateFormat = new SimpleDateFormat(format);java.util.Date dates = simpleDateFormat.parse(dateStr);date = new java.sql.Date(dates.getTime());} catch (ParseException e) {e.printStackTrace();}return date;}} 日期參數(shù)格式化配置?
3: 自定義指定請(qǐng)求前綴的Servlet(一個(gè)前臺(tái), 一個(gè)后臺(tái))
package com.yjy.test.game.web.servlet;import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.web.servlet.DispatcherServlet; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;/*** 后臺(tái)servlet* 需要添加對(duì)應(yīng)的 *-servlet.xml** @Author yjy* @Date 2018-04-23 16:26*/ @WebServlet(name = "backServlet", urlPatterns = {"/manager/admin/*"}) public class CustomBackServlet extends DispatcherServlet {private static final Logger log = LoggerFactory.getLogger(CustomBackServlet.class);@Overrideprotected void doService(HttpServletRequest request, HttpServletResponse response) throws Exception {log.info("backServlet doService...");super.doService(request, response);}} 后臺(tái)自定義Servlet package com.yjy.test.game.web.servlet;import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.web.servlet.DispatcherServlet; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;/*** 前臺(tái)servlet* 需要添加對(duì)應(yīng)的 *-servlet.xml** @Author yjy* @Date 2018-04-23 16:26*/ @WebServlet(name = "frontServlet", urlPatterns = {"/*"}) public class CustomFrontServlet extends DispatcherServlet {private static final Logger log = LoggerFactory.getLogger(CustomFrontServlet.class);@Overrideprotected void doService(HttpServletRequest request, HttpServletResponse response) throws Exception {log.info("frontServlet doService...");super.doService(request, response);}} 前臺(tái)自定義Servlet對(duì)應(yīng)的默認(rèn)xml文件, 打包war的時(shí)候需要, 看pom.xml中相應(yīng)配置, 否則到外部tomcat運(yùn)行時(shí), 會(huì)報(bào)找不到對(duì)應(yīng)的配置文件的錯(cuò)誤
xml內(nèi)容都是一樣的
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd"><!-- 此文件用于項(xiàng)目導(dǎo)出war包在外部tomcat運(yùn)行時(shí)檢測(cè), 如果沒有此文件, 則自定義Servlet無法訪問 --></beans> 自定義Servlet默認(rèn)配置?
? 4: 因?yàn)樯厦鎯蓚€(gè)自定義的Servlet繼承自DispatcherServlet, 不允許重寫init()方法, 所以如果需要自定義初始化ServletContext, 則必須自己寫一個(gè)Servlet繼承HttpServlet,(?此Servlet不需要配置相應(yīng)的xml文件)
package com.yjy.test.game.web.servlet;import com.yjy.test.game.service.OptionItemService; import org.springframework.web.context.WebApplicationContext; import org.springframework.web.context.support.WebApplicationContextUtils;import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet;/*** 自定義初始化 ServletContext** WebServlet中 urlPatterns必須填寫, 否則不會(huì)加載此Servlet, 同時(shí)需要配置 loadOnStartup = 1** @Author yjy* @Date 2018-05-02 11:47*/ @WebServlet(urlPatterns = "", loadOnStartup = 1) public class DictServlet extends HttpServlet {private OptionItemService optionItemService;public void setOptionItemService(OptionItemService optionItemService) {this.optionItemService = optionItemService;}public void init() throws ServletException {System.out.println("DictServlet init..............................");WebApplicationContext wac = WebApplicationContextUtils.getRequiredWebApplicationContext(this.getServletContext());setOptionItemService(wac.getBean (OptionItemService.class));optionItemService.getAllFieldName();// init something...// 例子: 設(shè)置Servlet全局屬性this.getServletContext().setAttribute("appName", "項(xiàng)目名");super.init();}} 初始化ServletContext?
5:?靜態(tài)資源請(qǐng)求配置
?
package com.yjy.test.config;import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.config.annotation.InterceptorRegistry; import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry; import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;/*** 自定義WebMvcConfigurerAdapter配置** @Author yjy* @Date 2018-04-23 11:40*/ @Configuration public class WebMvcConfig extends WebMvcConfigurerAdapter {private static final Logger log = LoggerFactory.getLogger(WebMvcConfig.class);/*** 靜態(tài)資源請(qǐng)求配置* @param registry 資源處理注冊(cè)器*/@Overridepublic void addResourceHandlers(ResourceHandlerRegistry registry) {log.info("addResourceHandlers...........................");registry.addResourceHandler("/**").addResourceLocations("classpath:/common/");super.addResourceHandlers(registry);}} 靜態(tài)資源配置?
6: tomcat上傳配置
package com.yjy.test.config;import org.springframework.boot.web.servlet.MultipartConfigFactory; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import javax.servlet.MultipartConfigElement;/*** 配置tomcat上傳限制** @Author yjy* @Date 2018-04-24 14:38*/ @Configuration public class MultipartConfig {/*** 配置tomcat上傳限制* @return 配置*/@Beanpublic MultipartConfigElement multipartConfigElement(){MultipartConfigFactory factory = new MultipartConfigFactory();factory.setMaxFileSize("50MB");factory.setMaxRequestSize("10MB");return factory.createMultipartConfig();}} 上傳配置?
7: 前后臺(tái)登入攔截器, 以及相應(yīng)配置類
package com.yjy.test.game.web.interceptor;import com.yjy.test.game.entity.Config; import com.yjy.test.game.entity.User; import com.yjy.test.game.service.ConfigService; import com.yjy.test.game.util.FrontUtils; import com.yjy.test.game.web.ErrorCode; import com.yjy.test.util.UnicodeUtil; import org.apache.commons.lang3.StringUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.PropertySource; import org.springframework.web.servlet.handler.HandlerInterceptorAdapter; import org.springframework.web.util.UrlPathHelper;import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.util.ArrayList; import java.util.Enumeration; import java.util.List;/*** 前臺(tái)登入攔截器** @Author yjy* @Date 2018-04-24 15:03*/ // 這里導(dǎo)入前綴為 front.login 的配置參數(shù) @ConfigurationProperties(prefix = "front.login") public class FrontLoginInterceptor extends HandlerInterceptorAdapter {private static final Logger log = LoggerFactory.getLogger(FrontLoginInterceptor.class);// 例外private List<String> excludeUrls = new ArrayList<>();private ConfigService configService;@Autowiredpublic void setConfigService(ConfigService configService) {this.configService = configService;}@Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object handler) throws Exception {log.info("FrontLoginInterceptor > excludeUrls: {}", excludeUrls);String uri = getURI(request);if (exclude(uri)) {return true;}try {request.setCharacterEncoding("UTF-8");response.setCharacterEncoding("UTF-8");response.setContentType("text/html;charset=UTF-8");User user = FrontUtils.getCurrentUser(request);if (null == user) {Enumeration s = request.getHeaderNames();String requestType = request.getHeader("X-Requested-With");if (requestType != null && requestType.equals("XMLHttpRequest")) {response.setCharacterEncoding("UTF-8");response.setContentType("application/json; charset=utf-8");response.getOutputStream().print("{\"status\":0,\"info\":\""+ UnicodeUtil.toEncodedUnicode( "登錄超時(shí),請(qǐng)重新登錄", false)+ "\", \"data\":null, \"code\": \"" + ErrorCode.ER_NOT_LOGIN + "\"}" );return false;}Config config = configService.findThisConfig();String path = null;if(null != config){path = StringUtils.isNotBlank(request.getContextPath()) ? request.getContextPath():"";}String reLogin = "/autho.jtk";if(StringUtils.isNotBlank(path) && path.length() > 1) {reLogin = path + reLogin;}response.sendRedirect(reLogin);return false;}} catch (Exception e) {log.error("檢查前臺(tái)登錄參數(shù)出錯(cuò)", e);}return super.preHandle(request, response, handler);}private boolean exclude(String uri) {if (excludeUrls != null) {for (String exc : excludeUrls) {if (exc.equals(uri)) {return true;}}}return false;}/*** 獲得第三個(gè)路徑分隔符的位置** @param request* @throws IllegalStateException* 訪問路徑錯(cuò)誤,沒有三(四)個(gè)'/'*/private static String getURI(HttpServletRequest request)throws IllegalStateException {UrlPathHelper helper = new UrlPathHelper();String uri = helper.getOriginatingRequestUri(request);return uri;}public List<String> getExcludeUrls() {return excludeUrls;}public void setExcludeUrls(List<String> excludeUrls) {this.excludeUrls = excludeUrls;}public ConfigService getConfigService() {return configService;} } 前臺(tái)登入攔截 package com.yjy.test.game.web.interceptor;import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;import com.yjy.test.game.entity.Admin; import com.yjy.test.game.entity.Config; import com.yjy.test.game.service.ConfigService; import com.yjy.test.game.util.BackUtils; import com.yjy.test.game.util.UnicodeUtil; import org.apache.commons.lang3.StringUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.web.servlet.handler.HandlerInterceptorAdapter; import org.springframework.web.util.UrlPathHelper;import java.util.ArrayList; import java.util.List;/*** 后臺(tái)上下文登錄檢測(cè)** @author wdy* @version :2016年2月29日 下午6:22:56*/ // 這里導(dǎo)入前綴為 back.login 的配置參數(shù) @ConfigurationProperties(prefix = "back.login") public class AdminLoginInterceptor extends HandlerInterceptorAdapter {private static final Logger log = LoggerFactory.getLogger(AdminLoginInterceptor.class);// 例外private List<String> excludeUrls = new ArrayList<>();private ConfigService configService;@Autowiredpublic void setConfigService(ConfigService configService) {this.configService = configService;}@Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object handler) throws Exception {String uri = getURI(request);if (exclude(uri)) {return true;}try {request.setCharacterEncoding("UTF-8");response.setCharacterEncoding("UTF-8");response.setContentType("text/html;charset=UTF-8");Admin user = BackUtils.getCurrentUser(request);if (null == user) {String requestType = request.getHeader("X-Requested-With");if (requestType != null && requestType.equals("XMLHttpRequest")) {response.setCharacterEncoding("UTF-8");response.setContentType("application/json; charset=utf-8");response.getOutputStream().print("{\"status\":2, \"code\":\"login\", \"info\":\""+ UnicodeUtil.toEncodedUnicode("登錄超時(shí),請(qǐng)重新登錄", false)+ "\", \"data\":null}");return false;}Config config = configService.findThisConfig();String path = null;if (null != config) {path = StringUtils.isNotBlank(request.getContextPath()) ? request.getContextPath() : "";}String reLogin = "/manager/admin/login.do";if (StringUtils.isNotBlank(path) && path.length() > 1) {reLogin = path + reLogin;}response.sendRedirect(reLogin);return false;}} catch (Exception e) {log.error("檢查后臺(tái)登錄參數(shù)出錯(cuò)", e);}return super.preHandle(request, response, handler);}private boolean exclude(String uri) {if (excludeUrls != null) {for (String exc : excludeUrls) {if (exc.equals(uri)) {return true;}}}return false;}/*** 獲得第三個(gè)路徑分隔符的位置** @param request* @throws IllegalStateException 訪問路徑錯(cuò)誤,沒有三(四)個(gè)'/'*/private static String getURI(HttpServletRequest request)throws IllegalStateException {UrlPathHelper helper = new UrlPathHelper();String uri = helper.getOriginatingRequestUri(request);String ctxPath = helper.getOriginatingContextPath(request);int start = 0, i = 0, count = 2;if (!StringUtils.isBlank(ctxPath)) {count++;}while (i < count && start != -1) {start = uri.indexOf('/', start + 1);i++;}if (start <= 0) {throw new IllegalStateException("admin access path not like '/manager/admin/...' pattern: "+ uri);}return uri.substring(start);}public List<String> getExcludeUrls() {return excludeUrls;}public void setExcludeUrls(List<String> excludeUrls) {this.excludeUrls = excludeUrls;}public ConfigService getConfigService() {return configService;} } 后臺(tái)登入攔截 package com.yjy.test.game.web.config;import com.yjy.test.game.web.interceptor.AdminLoginInterceptor; import com.yjy.test.game.web.interceptor.FrontLoginInterceptor; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.config.annotation.InterceptorRegistry; import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;/*** 自定義WebMvcConfigurerAdapter配置** @Author yjy* @Date 2018-04-23 11:40*/ @Configuration public class GameWebMvcConfig extends WebMvcConfigurerAdapter {private static final Logger log = LoggerFactory.getLogger(GameWebMvcConfig.class);/*** 攔截器配置* @param registry 攔截器注冊(cè)器*/@Overridepublic void addInterceptors(InterceptorRegistry registry) {log.info("addInterceptors1....................");registry.addInterceptor(getFrontLoginInterceptor()).addPathPatterns("*.jtk", "/*.jtk", "/*/*.jtk", "/*/*/*.jtk");registry.addInterceptor(getAdminLoginInterceptor()).addPathPatterns("*.do", "/*.do", "/*/*.do", "/*/*/*.do");super.addInterceptors(registry);}@BeanAdminLoginInterceptor getAdminLoginInterceptor() {return new AdminLoginInterceptor();}@BeanFrontLoginInterceptor getFrontLoginInterceptor() {return new FrontLoginInterceptor();}} 攔截器配置類
? 8: 添加過濾器
package com.yjy.test.game.web.filter;import java.io.IOException;import javax.servlet.Filter; import javax.servlet.FilterChain; import javax.servlet.FilterConfig; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import javax.servlet.annotation.WebFilter; import javax.servlet.http.HttpServletRequest;import org.slf4j.Logger; import org.slf4j.LoggerFactory;/*** 記錄請(qǐng)求執(zhí)行時(shí)間*/ @WebFilter(urlPatterns = "/*") public class ProcessTimeFilter implements Filter {protected final Logger log = LoggerFactory.getLogger(ProcessTimeFilter.class);/*** 請(qǐng)求執(zhí)行開始時(shí)間*/public static final String START_TIME = "_start_time";public void destroy() {}public void doFilter(ServletRequest req, ServletResponse response, FilterChain chain)throws IOException, ServletException {HttpServletRequest request = (HttpServletRequest) req;long time = System.currentTimeMillis();log.info("process start at {} for uri: {}", time, request.getRequestURI());request.setAttribute(START_TIME, time);chain.doFilter(request, response);time = System.currentTimeMillis() - time;log.info("process in {} ms: {}", time, request.getRequestURI());}public void init(FilterConfig arg0) throws ServletException {log.info("CustomFilter: ProcessTimeFilter init....");}} 請(qǐng)求執(zhí)行時(shí)間過濾器?
第六步: 遷移原項(xiàng)目源碼 幾個(gè)遇到問題的點(diǎn):
1: hibernate -> hibernate + JPA
原來的 *-hbm.xml 映射方式全部需要改成注解的方式, 實(shí)體類注解子如下:
package com.yjy.test.game.entity.club;import com.yjy.test.base.BaseEntity;import javax.persistence.*; import java.math.BigInteger; import java.util.Date;/*** 俱樂部消息表** @author yjy* Created on 2017年12月6日 上午9:34:07*/ @Entity @Table(name = "cg_club_message") public class ClubMessage extends BaseEntity {private static final long serialVersionUID = -1353909238958898740L;@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id; // idprivate Long receiveId; // 消息接收人private Long sendId; // 消息發(fā)送人private Long clubId; // 俱樂部idprivate Long clubUserId; // 相關(guān)成員idprivate Integer type; // 類型private Integer status; // 已操作/已讀狀態(tài)private Integer result; // 申請(qǐng)結(jié)果private String remark; // 備注private Integer isDelete; // 是否刪除private Date addTime; // 創(chuàng)建時(shí)間private Date updateTime; // 更新時(shí)間 @ManyToOne@JoinColumn(name = "clubId", insertable = false, updatable = false,foreignKey = @ForeignKey(name = "none", value = ConstraintMode.NO_CONSTRAINT ))private ClubUser clubUser;// 非持久化字段 @Transientprivate String nickName; // 昵稱 @Transientprivate String headImg; // 頭像 @Transientprivate String userCode; // 用戶code @Transientprivate String clubName; // 俱樂部名稱public ClubMessage() {}public ClubMessage(Long sendId, Long receiveId, Long clubId, Long clubUserId, Integer type) {this.sendId = sendId;this.receiveId = receiveId;this.clubId = clubId;this.clubUserId = clubUserId;this.type = type;this.init();}private void init() {this.isDelete = NO;}public Long getId() {return id;}public void setId(Long id) {this.id = id;}public Long getReceiveId() {return receiveId;}public void setReceiveId(Long receiveId) {this.receiveId = receiveId;}public Long getSendId() {return sendId;}public String getNickName() {return nickName;}public Long getClubId() {return clubId;}public Integer getIsDelete() {return isDelete;}public void setIsDelete(Integer isDelete) {this.isDelete = isDelete;}public void setClubId(Long clubId) {this.clubId = clubId;}public void setNickName(String nickName) {this.nickName = nickName;}public String getHeadImg() {return headImg;}public void setHeadImg(String headImg) {this.headImg = headImg;}public String getUserCode() {return userCode;}public void setUserCode(String userCode) {this.userCode = userCode;}public String getClubName() {return clubName;}public void setClubName(String clubName) {this.clubName = clubName;}public void setSendId(Long sendId) {this.sendId = sendId;}public Long getClubUserId() {return clubUserId;}public void setClubUserId(Long clubUserId) {this.clubUserId = clubUserId;}public ClubUser getClubUser() {return clubUser;}public void setClubUser(ClubUser clubUser) {this.clubUser = clubUser;}public Integer getType() {return type;}public void setType(Integer type) {this.type = type;}public Integer getStatus() {return status;}public void setStatus(Integer status) {this.status = status;}public Integer getResult() {return result;}public void setResult(Integer result) {this.result = result;}public String getRemark() {return remark;}public void setRemark(String remark) {this.remark = remark;}public Date getAddTime() {return addTime;}public void setAddTime(Date addTime) {this.addTime = addTime;}public Date getUpdateTime() {return updateTime;}public void setUpdateTime(Date updateTime) {this.updateTime = updateTime;}@Overridepublic String toString() {return "ClubMessageDao [id=" + id + ", receiveId=" + receiveId+ ", sendId=" + sendId + ", clubId=" + clubId + ", clubUserId="+ clubUserId + ", type=" + type + ", status=" + status+ ", result=" + result + ", remark=" + remark + ", isDelete="+ isDelete + ", addTime=" + addTime + ", updateTime="+ updateTime + ", nickName=" + nickName + ", headImg="+ headImg + ", userCode=" + userCode + ", clubName=" + clubName+ "]";}} 實(shí)體類例子 package com.yjy.test.base;import java.io.Serializable;/*** 實(shí)體類父類*/ public class BaseEntity extends BaseClass implements Serializable {} BaseEntity package com.yjy.test.base;import org.apache.commons.lang3.StringUtils;public abstract class BaseClass {protected static final int YES = 1;protected static final int NO = 0;/*** 驗(yàn)證字符串* @param s 字符串* @return 是否為空*/protected static boolean isBlank(String s) {return StringUtils.isBlank(s);}/*** 驗(yàn)證字符串* @param s 字符串* @return 是否不為空*/protected static boolean notBlank(String s) {return StringUtils.isNotBlank(s);}/*** 驗(yàn)證字符串* @param s 字符串* @return 是否數(shù)字*/protected static boolean isNumber(String s) {return StringUtils.isNumeric(s);}} BaseClass?
2: 重寫B(tài)ase層(代碼如下),注意: 原來BaseDaoImpl中的 sessionFactory 沒有了, 就是說不能通過getSession().createSQLQuery(sql) 的方式獲取SQLQuery了, 需要通過em.createNativeQuery(sql).unwrap(SQLQuery.class); 來獲得SQLQuery, em在BaseServiceImpl中已經(jīng)注入, 通過這種方式可以兼容之前的代碼
package com.yjy.test.base;import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.data.jpa.repository.Query; import org.springframework.data.repository.NoRepositoryBean;import java.io.Serializable;/*** BaseJpaRepository** @Author yjy* @Date 2018-04-25 15:55*/ @NoRepositoryBean public interface BaseJpaRepository<T extends BaseEntity, L extends Serializable> extends JpaRepository<T, L> {public T findTopByOrderByIdDesc();} BaseJpaRepository.java package com.yjy.test.base;import com.yjy.test.util.hibernate.Pagination; import org.springframework.data.domain.Sort;import java.io.Serializable; import java.util.List;public interface BaseService<T extends BaseEntity, L extends Serializable> {/*** 保存對(duì)象** @param entity 實(shí)體對(duì)象* @return 操作信息*/T save(T entity);T update(T entity);void delete(T entity);/*** 根據(jù)ID刪除記錄** @param id 記錄ID*/void deleteById(L id);/*** 根據(jù)ID數(shù)組刪除記錄,當(dāng)發(fā)生異常時(shí),操作終止并回滾** @param ids 記錄ID數(shù)組* @return 刪除的對(duì)象*/void deleteById(L[] ids);/*** 保存并刷新對(duì)象,避免many-to-one屬性不完整** @param entity*/T saveAndRefresh(T entity);/*** 通過ID查找對(duì)象** @param id 記錄的ID* @return 實(shí)體對(duì)象*/T findById(L id);T load(L id);T findByProperty(String property, Object value);List<T> findListByProperty(String property, Object value);/*** 根據(jù)屬性查找* @param propertyName 屬性* @param value 值* @param anywhere 是否模糊匹配* @return*/List<T> findListByProperty(String propertyName, Object value, boolean anywhere);/*** 查找所有對(duì)象** @return 對(duì)象列表*/List<T> findAll();/*** 分頁查詢* @param pageNo 頁號(hào)* @param pageSize 條數(shù)* @param orders 排序規(guī)則* @return 分頁列表*/Pagination findAllPage(int pageNo, int pageSize, Sort.Order... orders);List<T> findList(T entity, Sort.Order... orders);List<T> findList(T entity, int pageNo, int pageSize, Sort.Order... orders);Pagination findListPage(T entity, int pageNo, int pageSize, Sort.Order... orders);T findLast();T findFirst(T entity, Sort.Order... orders);long findAllCount();long findCount(T entity);} BaseService package com.yjy.test.base;import com.yjy.test.util.hibernate.Finder; import com.yjy.test.util.hibernate.Pagination; import org.hibernate.Query; import org.springframework.data.domain.*; import org.springframework.data.jpa.repository.JpaRepository; import javax.persistence.EntityManager; import javax.persistence.PersistenceContext; import javax.persistence.criteria.CriteriaBuilder; import javax.persistence.criteria.CriteriaQuery; import javax.persistence.criteria.Predicate; import javax.persistence.criteria.Root; import java.io.Serializable; import java.lang.reflect.ParameterizedType; import java.util.ArrayList; import java.util.List;// 這里不能加@Service注解, 否則會(huì)無法獲取泛型T的Class public class BaseServiceImpl<T extends BaseEntity, L extends Serializable>extends BaseClass implements BaseService<T, L> {//@Autowired和@PersistenceContext注解任取一 @PersistenceContextprotected EntityManager em;@Overridepublic T save(T entity) {return dao.save(entity);}@Overridepublic T update(T entity) {return dao.saveAndFlush(entity);}@Overridepublic void delete(T entity) {dao.delete(entity);}@Overridepublic void deleteById(L id) {dao.delete(id);}@Overridepublic void deleteById(L[] ids) {if (ids != null) {for (L id : ids) {dao.delete(id);}}}@Overridepublic T saveAndRefresh(T entity) {return dao.saveAndFlush(entity);}@Overridepublic T findById(L id) {return dao.findOne(id);}@Overridepublic T load(L id) {return dao.getOne(id);}@Overridepublic T findByProperty(String property, Object value) {List<T> list = findListByProperty(property, value);return list != null ? list.get(0) : null;}@Overridepublic List<T> findListByProperty(String property, Object value) {return findListByProperty(property, value, false);}@Overridepublic List<T> findListByProperty(String property, Object value, boolean anywhere) {CriteriaBuilder criteriaBuilder = em.getCriteriaBuilder();CriteriaQuery<T> query = criteriaBuilder.createQuery(getPersistentClass());Root<T> root = query.from(getPersistentClass());Predicate predicate;if (anywhere)predicate = criteriaBuilder.like(root.get(property), "%" + value.toString() + "%");elsepredicate = criteriaBuilder.equal(root.get(property), value);query.where(predicate);return em.createQuery(query).getResultList();}@Overridepublic List<T> findAll() {return dao.findAll();}@Overridepublic Pagination findAllPage(int pageNo, int pageSize, Sort.Order... orders) {Sort sort = orders != null && orders.length > 0 ? new Sort(orders) : null;Pageable pageable = new PageRequest(pageNo - 1, pageSize, sort);Page<T> page = dao.findAll(pageable);Pagination pagination = new Pagination(pageNo, pageSize, (int) page.getTotalElements());pagination.setList(page.getContent());return pagination;}@Overridepublic List<T> findList(T entity, Sort.Order... orders) {Example<T> example = Example.of(entity);if (orders != null && orders.length > 0)return dao.findAll(example, new Sort(orders));elsereturn dao.findAll(example);}@Override@SuppressWarnings("unchecked")public List<T> findList(T entity, int pageNo, int pageSize, Sort.Order... orders) {Pagination pagination = findListPage(entity, pageNo, pageSize, orders);if (pagination != null) {return (List<T>)pagination.getList();}return new ArrayList<>();}@Overridepublic Pagination findListPage(T entity, int pageNo, int pageSize, Sort.Order... orders) {Example<T> example = Example.of(entity);Sort sort = orders != null && orders.length > 0 ? new Sort(orders) : null;Pageable pageable = new PageRequest(pageNo - 1, pageSize, sort);Page<T> page = dao.findAll(example, pageable);Pagination pagination = new Pagination(pageNo, pageSize, (int) page.getTotalElements());pagination.setList(page.getContent());return pagination;}@Overridepublic T findLast() {return dao.findTopByOrderByIdDesc();}@Overridepublic T findFirst(T entity, Sort.Order... orders) {List<T> list = findList(entity, 1, 1, orders);if (!list.isEmpty()) {return list.get(0);}return null;}@Overridepublic long findAllCount() {return dao.count();}@Overridepublic long findCount(T entity) {Example<T> example = Example.of(entity);return dao.count(example);}@SuppressWarnings("rawtypes")protected Pagination find(Finder finder, int pageNo, int pageSize) {int totalCount = countQueryResult(finder);Pagination p = new Pagination(pageNo, pageSize, totalCount);if (totalCount < 1) {p.setList(new ArrayList());return p;}Query query = em.createQuery(finder.getOrigHql()).unwrap(Query.class);finder.setParamsToQuery(query);query.setFirstResult(p.getFirstResult());query.setMaxResults(p.getPageSize());List list = query.list();p.setList(list);return p;}/*** 通過count查詢獲得本次查詢所能獲得的對(duì)象總數(shù).** @param finder* @return*/protected int countQueryResult(Finder finder) {Query query = em.createQuery(finder.getRowCountHql()).unwrap(Query.class);finder.setParamsToQuery(query);return ((Number) query.iterate().next()).intValue();}/*************************************************************************/private Class<T> persistentClass;@SuppressWarnings("unchecked")public BaseServiceImpl() {ParameterizedType parameterizedType = (ParameterizedType) getClass().getGenericSuperclass();this.persistentClass = (Class<T>) parameterizedType.getActualTypeArguments()[0];}private BaseJpaRepository<T, L> dao;public void setDao(BaseJpaRepository<T, L> dao) {this.dao = dao;}protected BaseJpaRepository<T, L> getDao() {return this.dao;}private Class<T> getPersistentClass() {return persistentClass;}public void setPersistentClass(Class<T> persistentClass) {this.persistentClass = persistentClass;} } BaseServiceImpl package com.yjy.test.util.hibernate;/*** 分頁接口*/ public interface Paginable {/*** 總記錄數(shù)* * @return*/int getTotalCount();/*** 總頁數(shù)* * @return*/int getTotalPage();/*** 每頁記錄數(shù)* * @return*/int getPageSize();/*** 當(dāng)前頁號(hào)* * @return*/int getPageNo();/*** 是否第一頁* * @return*/boolean isFirstPage();/*** 是否最后一頁* * @return*/boolean isLastPage();/*** 返回下頁的頁號(hào)*/int getNextPage();/*** 返回上頁的頁號(hào)*/int getPrePage(); } Paginable package com.yjy.test.util.hibernate;import java.util.List;/*** 列表分頁。包含list屬性。*/ @SuppressWarnings("serial") public class Pagination extends SimplePage implements java.io.Serializable, Paginable {public Pagination() { }/*** 構(gòu)造器* * @param pageNo* 頁碼* @param pageSize* 每頁幾條數(shù)據(jù)* @param totalCount* 總共幾條數(shù)據(jù)*/public Pagination(int pageNo, int pageSize, int totalCount) {super(pageNo, pageSize, totalCount);}/*** 構(gòu)造器* * @param pageNo* 頁碼* @param pageSize* 每頁幾條數(shù)據(jù)* @param totalCount* 總共幾條數(shù)據(jù)* @param list* 分頁內(nèi)容*/public Pagination(int pageNo, int pageSize, int totalCount, List<?> list) {super(pageNo, pageSize, totalCount);this.list = list;}/*** 第一條數(shù)據(jù)位置** @return*/public int getFirstResult() {return (pageNo - 1) * pageSize;}/*** 當(dāng)前頁的數(shù)據(jù)*/private List<?> list;/*** 獲得分頁內(nèi)容* * @return*/public List<?> getList() {return list;}/*** 設(shè)置分頁內(nèi)容* * @param list*/@SuppressWarnings("rawtypes")public void setList(List list) {this.list = list;} } Pagination package com.yjy.test.util.hibernate;/*** 簡單分頁類*/ public class SimplePage implements Paginable {public static final int DEF_COUNT = 20;/*** 檢查頁碼 checkPageNo* * @param pageNo* @return if pageNo==null or pageNo<1 then return 1 else return pageNo*/public static int cpn(Integer pageNo) {return (pageNo == null || pageNo < 1) ? 1 : pageNo;}/*** 檢查每頁條數(shù)* @author yjy* Created on 2017年12月5日 下午2:39:23* @param pageSize 條數(shù)* @return 矯正值*/public static int cps(Integer pageSize) {return (pageSize == null || pageSize < 1) ? DEF_COUNT : pageSize;}/*** 根據(jù)頁號(hào)和條數(shù)計(jì)算起始下標(biāo)* @author yjy* Created on 2017年12月6日 上午9:36:17* @param pageNo 頁號(hào)* @param pageSize 條數(shù)* @return 起始下標(biāo) return (pageNo - 1) * pageSize*/public static int getStart(Integer pageNo, Integer pageSize) {return (cpn(pageNo) - 1) * cps(pageSize);}public SimplePage() {}/*** 構(gòu)造器* * @param pageNo* 頁碼* @param pageSize* 每頁幾條數(shù)據(jù)* @param totalCount* 總共幾條數(shù)據(jù)*/public SimplePage(int pageNo, int pageSize, int totalCount) {setTotalCount(totalCount);setPageSize(pageSize);setPageNo(pageNo);adjustPageNo();}/*** 調(diào)整頁碼,使不超過最大頁數(shù)*/public void adjustPageNo() {if (pageNo == 1) {return;}int tp = getTotalPage();if (pageNo > tp) {pageNo = tp;}}/*** 獲得頁碼*/public int getPageNo() {return pageNo;}/*** 每頁幾條數(shù)據(jù)*/public int getPageSize() {return pageSize;}/*** 總共幾條數(shù)據(jù)*/public int getTotalCount() {return totalCount;}/*** 總共幾頁*/public int getTotalPage() {int totalPage = totalCount / pageSize;if (totalPage == 0 || totalCount % pageSize != 0) {totalPage++;}return totalPage;}/*** 是否第一頁*/public boolean isFirstPage() {return pageNo <= 1;}/*** 是否最后一頁*/public boolean isLastPage() {return pageNo >= getTotalPage();}/*** 下一頁頁碼*/public int getNextPage() {if (isLastPage()) {return pageNo;} else {return pageNo + 1;}}/*** 上一頁頁碼*/public int getPrePage() {if (isFirstPage()) {return pageNo;} else {return pageNo - 1;}}protected int totalCount = 0;protected int pageSize = 20;protected int pageNo = 1;/*** if totalCount<0 then totalCount=0* * @param totalCount*/public void setTotalCount(int totalCount) {if (totalCount < 0) {this.totalCount = 0;} else {this.totalCount = totalCount;}}/*** if pageSize< 1 then pageSize=DEF_COUNT* * @param pageSize*/public void setPageSize(int pageSize) {if (pageSize < 1) {this.pageSize = DEF_COUNT;} else {this.pageSize = pageSize;}}/*** if pageNo < 1 then pageNo=1* * @param pageNo*/public void setPageNo(int pageNo) {if (pageNo < 1) {this.pageNo = 1;} else {this.pageNo = pageNo;}} } SimplePage3: 使用 mvn package 打包項(xiàng)目時(shí), 需要配置主類, 否則如果項(xiàng)目中存在多個(gè)類有main函數(shù)時(shí), 打包會(huì)報(bào)錯(cuò), 配置如下:
?
4: 有一個(gè)比較實(shí)用的點(diǎn), 就是可以通過@Value(property)注解將配置綁定至靜態(tài)變量中, 下面是Redis靜態(tài)配置類的例子:
package com.yjy.test.game.redis;import org.springframework.beans.factory.annotation.Value; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.context.annotation.PropertySource; import org.springframework.stereotype.Component; import org.springframework.validation.annotation.Validated;import javax.validation.constraints.NotNull;@Component @Validated public class RedisConfig {@NotNullpublic static String addr; //Redis服務(wù)器IP @NotNullpublic static int port; //Redis的端口號(hào)public static String auth; //訪問密碼 @NotNullpublic static int maxActive = 10; // 可用連接實(shí)例的最大數(shù)目,默認(rèn)值為8;如果賦值為-1,則表示不限制;// 如果pool已經(jīng)分配了maxActive個(gè)jedis實(shí)例,則此時(shí)pool的狀態(tài)為exhausted(耗盡)。 @NotNullpublic static int maxIdle = 200; //控制一個(gè)pool最多有多少個(gè)狀態(tài)為idle(空閑的)的jedis實(shí)例,默認(rèn)值也是8。 @NotNullpublic static int timeOut = 2000; //連接的超時(shí)時(shí)間 @NotNullpublic static int maxWait = 10000; //等待可用連接的最大時(shí)間,單位毫秒,默認(rèn)值為-1,表示永不超時(shí)。如果超過等待時(shí)間,則直接拋出JedisConnectionException; @NotNullpublic static boolean testOnBorrow = true; //在borrow一個(gè)jedis實(shí)例時(shí),是否提前進(jìn)行validate操作;如果為true,則得到的jedis實(shí)例均是可用的; @NotNullpublic static boolean testOnReturn = true; //在return一個(gè)jedis實(shí)例時(shí),是否提前進(jìn)行validate操作. @Value("${redis.addr}")public void setAddr(String addr) {this.addr = addr;}@Value("${redis.port}")public void setPort(int port) {this.port = port;}@Value("${redis.auth}")public void setAuth(String auth) {this.auth = auth;}@Value("${redis.maxActive}")public void setMaxActive(int maxActive) {this.maxActive = maxActive;}@Value("${redis.maxIdle}")public void setMaxIdle(int maxIdle) {this.maxIdle = maxIdle;}@Value("${redis.timeOut}")public void setTimeOut(int timeOut) {this.timeOut = timeOut;}@Value("${redis.maxWait}")public void setMaxWait(int maxWait) {this.maxWait = maxWait;}@Value("${redis.testOnBorrow}")public void setTestOnBorrow(boolean testOnBorrow) {this.testOnBorrow = testOnBorrow;}@Value("${redis.testOnReturn}")public void setTestOnReturn(boolean testOnReturn) {this.testOnReturn = testOnReturn;}public static void printAllConfig() {System.out.println("RedisConfig{" +"addr='" + addr + '\'' +", port=" + port +", auth='" + auth + '\'' +", maxActive=" + maxActive +", maxIdle=" + maxIdle +", timeOut=" + timeOut +", maxWait=" + maxWait +", testOnBorrow=" + testOnBorrow +", testOnReturn=" + testOnReturn +'}');}} RedisConfig.java?
大概就是這么個(gè)樣子!!! 嗯
?
?
轉(zhuǎn)載于:https://www.cnblogs.com/imyjy/p/8981345.html
總結(jié)
以上是生活随笔為你收集整理的SpringBoot webmvc项目导出war包并在外部tomcat运行产生的诸多问题以及解决方案的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Zookeeper源码分析(二) ---
- 下一篇: Mybatis中的jdbcType的作用