1 SpringBoot概述
敏捷开发(整合框架,直接加入依赖就好,自动配置);
无须Tomcat(Java应用程序方式运行,实际jar包),内置Tomcat
减少xml配置(甚至没有),配置文件properties/yml
注解形式
SpringBoot 是一个快速开发的框架,能够快速的整合第三方框架,简化XML配置,全部采用注解形式,内置Tomcat容器,帮助开发者能够实现快速开发,SpringBoot的Web组件 默认集成的是SpringMVC框架。SpringMVC是控制层。
1.1 SpringBoot和SpringMVC区别
SpringBoot 是一个快速开发的框架,能够快速的整合第三方框架,简化XML配置,全部采用注解形式,内置Tomcat容器,帮助开发者能够实现快速开发,SpringBoot的Web组件 默认集成的是SpringMVC框架。
SpringMVC是控制层。
1.2 SpringBoot和SpringCloud区别
SpringBoot 是一个快速开发的框架,能够快速的整合第三方框架,简化XML配置,全部采用注解形式,内置Tomcat容器,帮助开发者能够实现快速开发,SpringBoot的Web组件 默认集成的是SpringMVC框架。
SpringMVC是控制层。
SpringCloud依赖与SpringBoot组件,使用SpringMVC编写Http协议接口,同时SpringCloud是一套完整的微服务解决框架。
2 快速入门
依赖:
<!--提供dependency management,也就是说依赖管理,引入以后在申明其它dependency的时候就不需要version了,后面可以看到。--> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.0.2.RELEASE</version> <relativePath/> <!-- lookup parent from repository --> </parent> <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding> <java.version>1.8</java.version> </properties> <dependencies><!--springweb 核心组件--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> </dependencies> <!--如果我们要直接Main启动spring,那么以下plugin必须要添加,否则是无法启动的。如果使用maven 的spring-boot:run的话是不需要此配置的。--> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build>
后台程序:
@RestController@EnableAutoConfigurationpublic class HelloController { @RequestMapping("/hello") public String index() { return "Hello World"; } public static void main(String[] args) { SpringApplication.run(HelloController.class, args); }}
启动main主程序,访问http://127.0.0.1:8080/index看看是否有输出。
2.1 @EnableAutoConfiguration
Spring Boot 根据应用所声明的依赖来对 Spring 框架进行自动配置。这个注解告诉Spring Boot根据添加的jar依赖猜测你想如何配置Spring。由于spring-boot-starter-web添加了Tomcat和Spring MVC,所以auto-configuration将假定你正在开发一个web应用并相应地对Spring进行设置。但是这个注解仅仅扫描当前类声明的bean,其他类的bean不会注册。
2.2 @RestController
在类上加上RestController 表示该类所有的方法返回JSON格式,直接可以编写Restful接口
2.3 SpringBoot启动方式2
@ComponentScan(basePackages = "com.itmayiedu.controller")---控制器扫包范围
@ComponentScan(basePackages = "com.itboy.spboot.controller")@EnableAutoConfigurationpublic class App { public static void main(String[] args) { SpringApplication.run(App.class, args); }}
2.3 springboot启动方式3
@SpringBootApplication 被 @Configuration、@EnableAutoConfiguration、@ComponentScan 注解所修饰,换言之 Springboot 提供了统一的注解来替代以上三个注解
扫包范围:在启动类上加上@SpringBootApplication注解,当前包下或者子包下所有的类都可以扫到。
@SpringBootApplicationpublic class SpbootApplication { public static void main(String[] args) { SpringApplication.run(SpbootApplication.class, args); }}
3 web开发
3.1 静态资源访问
开发Web应用的时候,需要引用大量的js、css、图片等静态资源。
Spring Boot默认提供静态资源目录位置需置于resources下,目录名需符合如下规则:
/static
/public
/resources
/META-INF/resources
在src/main/resources/目录下创建static,在该位置放置一个图片文件。启动程序后,尝试访问http://localhost:8080/D.jpg。如能显示图片,配置成功。
3.2 渲染web界面
在之前的示例中,都是通过@RestController来处理请求,所以返回的内容为json对象。那么如果需要渲染html页面的时候,要如何实现呢?
模板引擎
在动态HTML实现上Spring Boot依然可以完美胜任,并且提供了多种模板引擎的默认配置支持,所以在推荐的模板引擎下,我们可以很快的上手开发动态网站。
Spring Boot提供了默认配置的模板引擎主要有以下几种:
- Thymeleaf
- FreeMarker
- Velocity
- Groovy
- Mustache
Spring Boot建议使用这些模板引擎,避免使用JSP,若一定要使用JSP将无法实现Spring Boot的多种特性,具体可见后文:支持JSP的配置
当你使用上述模板引擎中的任何一个,它们默认的模板配置路径为:src/main/resources/templates。当然也可以修改这个路径,具体如何修改,可在后续各模板引擎的配置属性中查询并修改。
3.2.1 使用Freemarker模板引擎渲染web视图
依赖:
<!-- 引入freeMarker的依赖包. --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-freemarker</artifactId></dependency>
后台代码,返回“index”表示到src/main/resources/templates路径下寻找index.ftl的伪静态页面文件,然后根据ModelAndView渲染显示
@RequestMapping("/index") public String index(Map<String, Object> map) { map.put("name","美丽的天使..."); return "index"; }
前台:
<!DOCTYPE html><html><head lang="en"><meta charset="UTF-8" /><title></title></head><body> ${name}</body> </html>
补充:
@RequestMapping("/freemarkerIndex") public String index(Map<String, Object> result) { result.put("name", "小明"); result.put("sex", "0"); List<String> listResult = new ArrayList<String>(); listResult.add("zhangsan"); listResult.add("lisi"); listResult.add("李大爷"); result.put("listResult", listResult); return "index"; } <!DOCTYPE html><html><head lang="en"><meta charset="UTF-8" /><title>首页</title></head><body> ${name}<#if sex=="1"> 男 <#elseif sex=="2"> 女 <#else> 其他 </#if> <#list userlist as user> ${user} </#list></body> </html>
freemarker的一些基本配置,配置在application.properties
###########################################################FREEMARKER (FreeMarkerAutoConfiguration)########################################################spring.freemarker.allow-request-override=falsespring.freemarker.cache=truespring.freemarker.check-template-location=truespring.freemarker.charset=UTF-8spring.freemarker.content-type=text/htmlspring.freemarker.expose-request-attributes=falsespring.freemarker.expose-session-attributes=falsespring.freemarker.expose-spring-macro-helpers=false#spring.freemarker.prefix=#spring.freemarker.request-context-attribute=#spring.freemarker.settings.*=spring.freemarker.suffix=.ftlspring.freemarker.template-loader-path=classpath:/templates/#comma-separated list#spring.freemarker.view-names= # whitelist of view names that can be resolved
3.2.2 使用jsp渲染web视图
依赖,因为Spring boot内嵌Tomcat不支持jsp,所以需要添加外部Tomcat支持
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.0.0.RELEASE</version> </parent> <dependencies> <!-- SpringBoot web 核心组件 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </dependency> <!-- SpringBoot 外部tomcat支持 --> <dependency> <groupId>org.apache.tomcat.embed</groupId> <artifactId>tomcat-embed-jasper</artifactId> </dependency> </dependencies>
application.properties配置读取jsp的目录,jsp文件不要放在resources目录下,因为这个目录会被打包然后是classpath,所以只能重新在main下新建webapp/WEB..不然访问不到。
spring.mvc.view.prefix=/WEB-INF/jsp/spring.mvc.view.suffix=.jsp
后台代码,和freemarker一样,只不过文件后缀名为application.properties指定的jsp
@Controllerpublic class IndexController { @RequestMapping("/index") public String index() { return "index"; }}
注意:创建SpringBoot整合JSP,一定要为war类型,否则会找不到页面.不要把JSP页面存放在resources// jsp 不能被访问到
3.3 全局捕获异常
@ExceptionHandler 表示拦截异常
- @ControllerAdvice 是 controller 的一个辅助类,最常用的就是作为全局异常处理的切面类
- @ControllerAdvice 可以指定扫描范围
- @ControllerAdvice 约定了几种可行的返回值,如果是直接返回 model 类的话,需要使用 @ResponseBody 进行 json 转换
- 返回 String,表示跳到某个 view
- 返回 modelAndView
- 返回 model + @ResponseBody
@ControllerAdvice(basePackages = {"com.itboy.spboot.controller"})public class GlobalExceptionHandler { @ExceptionHandler(RuntimeException.class) @ResponseBody public Map<String,Object> error(){ Map<String,Object> map=new HashMap<>(); map.put("errorCode","500"); map.put("errorMsg","系统错误!"); return map; }}
4 日志管理
4.1 使用log4j记录日志
依赖:
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.0.0.RELEASE</version> </parent> <dependencies> <!-- SpringBoot 核心组件 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </dependency> <dependency> <groupId>org.apache.tomcat.embed</groupId> <artifactId>tomcat-embed-jasper</artifactId> </dependency> <!-- spring boot start --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter</artifactId> <exclusions> <!-- 排除自带的logback依赖 --> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-logging</artifactId> </exclusion> </exclusions> </dependency> <!-- springboot-log4j --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-log4j</artifactId> <version>1.3.8.RELEASE</version> </dependency> </dependencies>
新建log4配置文件log4j.properties:
#log4j.rootLogger=CONSOLE,info,error,DEBUGlog4j.rootLogger=info,error,CONSOLE,DEBUGlog4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout log4j.appender.CONSOLE.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n log4j.logger.info=infolog4j.appender.info=org.apache.log4j.DailyRollingFileAppenderlog4j.appender.info.layout=org.apache.log4j.PatternLayout log4j.appender.info.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n log4j.appender.info.datePattern='.'yyyy-MM-ddlog4j.appender.info.Threshold = info log4j.appender.info.append=true #log4j.appender.info.File=/home/admin/pms-api-services/logs/info/api_services_infolog4j.appender.info.File=/Users/dddd/Documents/testspace/pms-api-services/logs/info/api_services_infolog4j.logger.error=error log4j.appender.error=org.apache.log4j.DailyRollingFileAppenderlog4j.appender.error.layout=org.apache.log4j.PatternLayout log4j.appender.error.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n log4j.appender.error.datePattern='.'yyyy-MM-ddlog4j.appender.error.Threshold = error log4j.appender.error.append=true #log4j.appender.error.File=/home/admin/pms-api-services/logs/error/api_services_errorlog4j.appender.error.File=/Users/dddd/Documents/testspace/pms-api-services/logs/error/api_services_errorlog4j.logger.DEBUG=DEBUGlog4j.appender.DEBUG=org.apache.log4j.DailyRollingFileAppenderlog4j.appender.DEBUG.layout=org.apache.log4j.PatternLayout log4j.appender.DEBUG.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n log4j.appender.DEBUG.datePattern='.'yyyy-MM-ddlog4j.appender.DEBUG.Threshold = DEBUG log4j.appender.DEBUG.append=true #log4j.appender.DEBUG.File=/home/admin/pms-api-services/logs/debug/api_services_debuglog4j.appender.DEBUG.File=/Users/dddd/Documents/testspace/pms-api-services/logs/debug/api_services_debug
通过在类中这样获取private static final Logger logger = LoggerFactory.getLogger(IndexController.class);使用
4.2 使用AOP统一处理web请求日志
依赖:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency>
aop:
@Aspect@Componentpublic class WebLogApect { private static final Logger logger=LoggerFactory.getLogger(WebLogApect.class); @Pointcut("execution(public * com.itboy.spboot.controller.*.*(..))") public void webLog(){ } @Before("webLog()") public void doBefore(JoinPoint joinPoint){ //接受到请求,记录请求内容 ServletRequestAttributes attributes= (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); HttpServletRequest request=attributes.getRequest(); logger.info("URL :"+request.getRequestURI().toString()); logger.info("HTTP_METHOD: "+request.getMethod()); logger.info("IP: "+request.getRemoteAddr()); Enumeration<String> eu=request.getParameterNames(); while (eu.hasMoreElements()){ String name=eu.nextElement(); logger.info("name:{},value{}",name,request.getParameter(name)); } } @AfterReturning(returning = "ret", pointcut = "webLog()") public void doAfterReturning(Object ret){ logger.info("RESPONSE :"+ret); }}
此时任意访问一个URL都会记录在日志文件中。
4.3 lombok
底层通过asm框架修改字节码文件,生成相应的方法。
注意需要在开发工具中先安装lombok插件,然后引入依赖:
<dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId></dependency>
例子:
@Slf4j@Datapublic class UserEntity { // @Getter // @Setter private String userName; // @Getter // @Setter private Integer age; @Override public String toString() { return "UserEntity [userName=" + userName + ", age=" + age + "]"; } public static void main(String[] args) { UserEntity userEntity = new UserEntity(); userEntity.setUserName("zhangsan"); userEntity.setAge(20); System.out.println(userEntity.toString()); log.info("####我是日志##########"); } }
其他特性:
@Data 标签,生成getter/setter toString()等方法 @NonNull : 让你不在担忧并且爱上NullPointerException @CleanUp : 自动资源管理:不用再在finally中添加资源的close方法 @Setter/@Getter : 自动生成set/get方法 @ToString : 自动生成toString方法 @EqualsAndHashcode : 从对象的字段中生成hashCode和equals的实现 @NoArgsConstructor/@RequiredArgsConstructor/@AllArgsConstructor 自动生成构造方法 @Data : 自动生成set/get方法,toString方法,equals方法,hashCode方法,不带参数的构造方法 @Value : 用于注解final类 @Builder : 产生复杂的构建器api类@SneakyThrows : 异常处理(谨慎使用) @Synchronized : 同步方法安全的转化 @Getter(lazy=true) : @Log : 支持各种logger对象,使用时用对应的注解,如:@Log4
5 简单但好用一些内容
5.1 使用@Scheduled创建定时任务
在Spring Boot主类中添加@EnableScheduling注解,启动定时任务的配置
然后声明定时任务:
@Component@Slf4jpublic class ScheduledTasks { private static final SimpleDateFormat dateFormat=new SimpleDateFormat("HH:mm:ss"); //周期5秒 @Scheduled(fixedRate = 5000) public void reportCurrentTime(){ log.info("现在时间:"+dateFormat.format(new Date())); }}
5.2 使用@Async实现异步调用
在启动主类上加上#EnableAsync,需要执行方法上加入@Async后,底层就会使用多线程技术
IndexController .java @RestController//表示该类所有方法返回json格式,等价于@Controller@ResponseBody@Slf4jpublic class IndexController { @Autowired private IndexService indexService; @RequestMapping("/async") public String asyncTest(){ log.info("##01##"); indexService.userThread(); log.info("##04##"); return "async success"; } } IndexService.java@Service@Slf4jpublic class IndexService { @Async//相当于这个方法重新开辟了一个单独的线程执行 //思路:使用aop技术,在运行时创建一个单独的线程执行 public void userThread(){ log.info("##02##"); try{ Thread.sleep(5*1000); } catch (Exception e){ } log.info("##03##");// 加上注解后类似下面这样执行// new Thread(new Runnable() {// @Override// public void run() {// log.info("##02##");// try{// Thread.sleep(5*1000);// } catch (Exception e){ }// log.info("##03##");// }// }).start(); }}
本来是要输出1234的,但是异步之后,结果是1423,所以采用了异步
5.3 @Value自定义参数
配置文件值
name=xiaoming.com
初始化时,就去配置文件读取该配置
@Value("${name}") private String name;@ResponseBody @RequestMapping("/getValue") public String getValue() { return name; }
5.4 多环境配置
application.properties:
spring.profiles.active=prod#test#dev三个表名使用哪个配置文件,也就是什么环境
下面三个配置文件表示不同的环境。
application-dev.properties开发环境
application-test.properties测试环境
application-prod.properties生产环境
5.5 修改端口号和项目路径
server.port=8080server.servlet.context-path=/spboot
5.6 Spring Boot yml使用
SpringBoot 默认读取 application.yml|properties
server: port: 8080 servlet: context-path: /
5.7 发布打包
打包方式指定?
<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.itboy</groupId> <artifactId>spboot</artifactId> <version>0.0.1-SNAPSHOT</version> <packaging>war或者jar选择一种</packaging>
5.7.1 Jar类型打包方式(打包后后缀名.jar)
1.使用mvn celan package 打包
2.使用java –jar 包名
5.7.2 war类型打包方式(打包后后缀名.war)
1.使用mvn celan package 打包
2.使用java –jar 包名
5.7.3 外部Tomcat运行
1.pom.xml
<!-- 打包类型--> <packaging>war</packaging> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build>
1.启动类重写configure的方法
public class SpbootApplication extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) { // TODO Auto-generated method stub return builder.sources(SpbootApplication.class); } public static void main(String[] args) { SpringApplication.run(SpbootApplication.class, args); }}
1.使用mvn celan package 打包
2. 将war包 放入到tomcat webapps下运行即可。
http://localhost:8080/spboot-0.0.1-SNAPSHOT/indexDev其中spboot-0.0.1-SNAPSHOT。
为包名
5.7.4 打包常见错误
代表你打包的时候没有指定主函数,在pom.xml文件中添加以下内容:
<build> <plugins> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-compiler-plugin</artifactId> <configuration> <source>1.8</source> <target>1.8</target> </configuration> </plugin> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <mainClass>com.itboy.spboot.SpbootApplication</mainClass> </configuration> <executions> <execution> <goals> <goal>repackage</goal> </goals> </execution> </executions> </plugin> </plugins> </build>
5.8 Spring Boot整合拦截器
拦截器是AOP的一种实现,底层通过动态代理实现。
与过滤器区别:
(1)拦截器是基于Java反射机制,过滤器基于函数回调
(2)拦截器不依赖于Servlet容器,过滤器依赖于Servlet
(3)拦截器只能对controller请求起作用,而过滤器可以几乎对所有的请求起作用
(4)在controller的生命周期中,拦截器可以多次被调用,过滤器只能在容器初始化时被调用一次
场景:1、过滤器常用于设置编码字符、过滤敏感字符2、拦截器常用于拦截未登录用户、审计日志
创建拦截器:
@Slf4j@Componentpublic class LoginInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws IOException { log.info("开始拦截登录请求"); String token=request.getParameter("token"); if(StringUtils.isEmpty(token)){ response.getWriter().write("not found token"); return false; } return true; }}
注册拦截器:
@Configurationpublic class WebAppConfig { @Autowired private LoginInterceptor loginInterceptor; @Bean public WebMvcConfigurer webMvcConfigurer(){ return new WebMvcConfigurer() { @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(loginInterceptor).addPathPatterns("/*"); } }; }}
6 数据访问
6.1 springboot整合使用JdbcTemplate
添加依赖:
<!--JdbcTemplate依赖--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jdbc</artifactId> </dependency> <!--mysql--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> </dependency>
配置文件.yml:
spring: datasource: url: jdbc:mysql://localhost:3306/test username: root password: root driver-class-name: com.mysql.jdbc.Driver
IndexController.java@RestController//表示该类所有方法返回json格式,等价于@Controller@ResponseBody@Slf4jpublic class IndexController { @Autowired private IndexService indexService; @RequestMapping("/createUser") public String createUser(){ int xiaoming = indexService.createUser("xiaoming", 23); return "success:"+xiaoming; } } IndexService.java@Service@Slf4jpublic class IndexService { @Autowired private JdbcTemplate jdbcTemplate; public int createUser(String name,Integer age){ int update = jdbcTemplate.update("insert into users (name,age) values (?,?);", name, age); return update; }
访问http://127.0.0.1:8080/createUser查看数据库是否创建成功以及页面是否正确响应。
6.2 Spring boot整合使用mybatis
添加依赖:
<!--mybatis集成依赖--> <dependency> <groupId>org.mybatis.spring.boot</groupId> <artifactId>mybatis-spring-boot-starter</artifactId> <version>1.1.1</version> </dependency>
实体类:
@Datapublic class User { private int id; private String name; private Integer age;}
mapper类:
@Mapperpublic interface UserMapper { //查询 @Select("SELECT * FROM USERS WHERE NAME=#{name}") User findByName(@Param("name") String name); //添加 @Insert("INSERT INTO USERS(NAME,AGE) VALUES (#{name},#{age})") int insert(@Param("name")String name,@Param("age")Integer age);}
service类:
@Servicepublic class UserService { @Autowired private UserMapper userMapper; public User findByName(String name){ return userMapper.findByName(name); } //添加 public int insert(String name,Integer age){ return userMapper.insert(name,age); } }
controller类:
@RestController//表示该类所有方法返回json格式,等价于@Controller@ResponseBody@Slf4jpublic class IndexController { @Autowired private UserService userService; @RequestMapping("/getUser") public User findByName(@RequestParam("name") String name){ return userService.findByName(name); } @RequestMapping("/addUser") public int addUser(@RequestParam("name") String name,@RequestParam("age") Integer age){ return userService.insert(name,age); } }
访问http://127.0.0.1:8080/addUser?name=xiaogou&age=23后再访问http://127.0.0.1:8080/getUser?name=xiaogou查看两次请求是否正确;
注意如果不想再mapper类中添加@Mapper注解,可以在启动类中天@MapperScan指定扫mapper包范围,多个以逗号隔开类似ComponentScan。两处必有一处需要添加。
6.3 mybatis整合分页插件pageHelper
PageHelper 是一款好用的开源免费的 Mybatis 第三方物理分页插件物理分页
支持常见的 12 种数据库。Oracle,MySql,MariaDB,SQLite,DB2,PostgreSQL,SqlServer 等
支持多种分页方式
支持常见的 RowBounds(PageRowBounds),PageHelper.startPage 方法调用,Mapper 接口参数调用
添加依赖:
<!--Spring boot整合pageHelper--> <dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pagehelper-spring-boot-starter</artifactId> <version>1.2.5</version> </dependency>
配置文件application.yml:
server: port: 8080 servlet: context-path: /spring: datasource: url: jdbc:mysql://localhost:3306/test username: root password: root driver-class-name: com.mysql.jdbc.Driver logging.level.com.example.demo.dao: DEBUGpagehelper: helper-dialect: mysql reasonable: true support-methods-arguments: true params: count=countSql page-size-zero: true
mapper:
@Mapperpublic interface UserMapper { //列表查询 @Select("SELECT * FROM USERS") List<User> findUserList();}
service:
@Servicepublic class UserService { @Autowired private UserMapper userMapper; //分页列表 public List<User> findUserList(int page,int pageSize){ //开启分页插件,放在查询语句上面 PageHelper.startPage(page,pageSize);//因为不同数据库的分页语句不同,所以底层是通过改写SQL语句来实现分页的 List<User> userList=userMapper.findUserList();//通过debug此时这里其实也只有两条数据 //封装分页之后的数据 PageInfo<User> userPageInfo=new PageInfo<>(userList);//这样只是返回的数据中包含一些分页参数而已,一般都是这种返回较好,这里为了简便 return userList; } }
controller:
@RestController@Slf4jpublic class IndexController { @Autowired private UserService userService; @RequestMapping("/listUsers") public List<User> listUser(@RequestParam("page") Integer page, @RequestParam("pageSize") Integer pageSize){ return userService.findUserList(page,pageSize); } }
访问http://127.0.0.1:8080/listUsers?page=1&pageSize=2查看结果
6.4 springboot 整合使用Springjpa
这个东东其实就是通过封装hibernate得来的
添加依赖:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-jpa</artifactId> </dependency>
实体类:
@Data@Entity(name="users")//name表示数据库中对应的表名,默认为类名首字母小写,其他大写字母为下划线public class UserEntity { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; @Column(name="name")//name=数据库中对应的列名,不写默认为成员变量名称,其他大写字母为下划线 private String name; @Column(name = "age") private Integer age;}
dao层:
public interface UserEntityRepository extends JpaRepository<UserEntity,Integer> { }
controller:
@RestController@Slf4jpublic class IndexController { @Autowired private UserEntityRepository userEntityRepository; @RequestMapping("/jpaFindUser") public Object japIndex(UserEntity userEntity){ Optional<UserEntity> userEntity1=userEntityRepository.findById(userEntity.getId()); UserEntity userEntity2=userEntity1.get(); return userEntity2==null?"没有找到用户":userEntity2; }}
访问http://127.0.0.1:8080/jpaFindUser?id=25检查是否正确。
6.5 Spring Boot整合多数据源
在一个项目中有多个数据源(连接不同数据库jdbc),一般来说这个连接数是无线大的,不过具体还是根据主机内存大小来设定合适大小较好。
那我怎么划分呢?一般有两种:分包名(业务不同包名肯定不同嘛) | 注解方式(有点类似于mybatis中的多数据源,通过注解选择相应的数据源)。
这里主要以分包方式来实践一下。
主要依赖:
<dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </dependency> <!--JdbcTemplate依赖--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jdbc</artifactId> </dependency> <!--mysql--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> </dependency> <!--mybatis集成依赖--> <dependency> <groupId>org.mybatis.spring.boot</groupId> <artifactId>mybatis-spring-boot-starter</artifactId> <version>1.1.1</version> </dependency>
配置文件:
spring: datasource: #数据源1 test2: jdbc-url: jdbc:mysql://localhost:3306/shiro?useSSL=false username: root password: root driver-class-name: com.mysql.jdbc.Driver #数据源2 test1: jdbc-url: jdbc:mysql://localhost:3306/test?useSSL=false username: root password: root driver-class-name: com.mysql.jdbc.Driver
两个数据库shiro和test并且都有一个users表。
实体类,分别在test1包和test2都有
@Datapublic class User { private int id; private String name; private Integer age;}
mapper,分别在test1和test2下都有
public interface UserMapper2 { //查询 @Select("SELECT * FROM USERS WHERE NAME=#{name}") User findByName(@Param("name") String name); //添加 @Insert("INSERT INTO USERS(NAME,AGE) VALUES (#{name},#{age})") int insert(@Param("name") String name, @Param("age") Integer age); //列表查询 @Select("SELECT * FROM USERS") List<User> findUserList();}
service:也是test1和test2都有,这里仅展示一个
@Servicepublic class UserService2 { @Autowired private UserMapper2 userMapper; public User findByName(String name){ return userMapper.findByName(name); } //添加 public int insert(String name,Integer age){ return userMapper.insert(name,age); } //分页列表 public List<User> findUserList(int page,int pageSize){ //开启分页插件,放在查询语句上面 PageHelper.startPage(page,pageSize);//因为不同数据库的分页语句不同,所以底层是通过改写SQL语句来实现分页的 List<User> userList=userMapper.findUserList();//通过debug此时这里其实也只有两条数据 //封装分页之后的数据 PageInfo<User> userPageInfo=new PageInfo<>(userList);//这样只是返回的数据中包含一些分页参数而已,一般都是这种返回较好,这里为了简便 return userList; } }
数据源1配置文件:
package com.itboy.spboot.datasource; import org.apache.ibatis.session.SqlSessionFactory;import org.mybatis.spring.SqlSessionFactoryBean;import org.mybatis.spring.SqlSessionTemplate;import org.mybatis.spring.annotation.MapperScan;import org.springframework.beans.factory.annotation.Qualifier;import org.springframework.boot.context.properties.ConfigurationProperties;import org.springframework.boot.jdbc.DataSourceBuilder;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.jdbc.datasource.DataSourceTransactionManager; import javax.sql.DataSource; /** * @ClassName: Datasource1Config * @Description: 数据源1 * @Version: v1.0 * @Copyright: ilose自主开发 **/@Configuration@MapperScan(basePackages = "com.itboy.spboot.test1.mapper",sqlSessionFactoryRef = "test1SqlSessionFactory")public class Datasource1Config { /** * 数据源1配置信息 * @return */ @Bean(name = "test1DataSource") @ConfigurationProperties(prefix = "spring.datasource.test1") public DataSource testDataSource(){ return DataSourceBuilder.create().build(); } /** * 数据源1会话工厂 * @param dataSource * @return * @throws Exception */ @Bean(name = "test1SqlSessionFactory") public SqlSessionFactory testSqlSessionFactory(@Qualifier("test1DataSource" ) DataSource dataSource) throws Exception { SqlSessionFactoryBean bean=new SqlSessionFactoryBean(); bean.setDataSource(dataSource);// bean.setMapperLocations(// new PathMatchingResourcePatternResolver().getResources("classpath:mybatis/mapper/test1/*.xml")// ); return bean.getObject(); } /** * 数据源1事务管理 * @param dataSource * @return */ @Bean(name = "test1TransactionManager") public DataSourceTransactionManager testDataSourceTransactionManager(@Qualifier("test1DataSource") DataSource dataSource){ return new DataSourceTransactionManager(dataSource); } /** * * @param sqlSessionFactory * @return */ @Bean(name = "test1SqlSessionTemplate") public SqlSessionTemplate testSqlSessionTemplate(@Qualifier("test1SqlSessionFactory") SqlSessionFactory sqlSessionFactory){ return new SqlSessionTemplate(sqlSessionFactory); } }
数据源2配置文件:
package com.itboy.spboot.datasource; import org.apache.ibatis.session.SqlSessionFactory;import org.mybatis.spring.SqlSessionFactoryBean;import org.mybatis.spring.SqlSessionTemplate;import org.mybatis.spring.annotation.MapperScan;import org.springframework.beans.factory.annotation.Qualifier;import org.springframework.boot.context.properties.ConfigurationProperties;import org.springframework.boot.jdbc.DataSourceBuilder;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.jdbc.datasource.DataSourceTransactionManager; import javax.sql.DataSource; /** * @ClassName: Datasource1Config * @Description: 数据源1 * @Version: v1.0 * @Copyright: ilose自主开发 **/@Configuration@MapperScan(basePackages = "com.itboy.spboot.test2.mapper",sqlSessionFactoryRef = "test2SqlSessionFactory")public class Datasource2Config { /** * 数据源2配置信息 * @return */ @Bean(name = "test2DataSource") @ConfigurationProperties(prefix = "spring.datasource.test2") public DataSource testDataSource(){ return DataSourceBuilder.create().build(); } /** * 数据源2会话工厂 * @param dataSource * @return * @throws Exception */ @Bean(name = "test2SqlSessionFactory") public SqlSessionFactory testSqlSessionFactory(@Qualifier("test2DataSource" ) DataSource dataSource) throws Exception { SqlSessionFactoryBean bean=new SqlSessionFactoryBean(); bean.setDataSource(dataSource);// bean.setMapperLocations(// new PathMatchingResourcePatternResolver().getResources("classpath:mybatis/mapper/test1/*.xml")// ); return bean.getObject(); } /** * 数据源1事务管理 * @param dataSource * @return */ @Bean(name = "test2TransactionManager") public DataSourceTransactionManager testDataSourceTransactionManager(@Qualifier("test2DataSource") DataSource dataSource){ return new DataSourceTransactionManager(dataSource); } /** * * @param sqlSessionFactory * @return */ @Bean(name = "test2SqlSessionTemplate") public SqlSessionTemplate testSqlSessionTemplate(@Qualifier("test2SqlSessionFactory") SqlSessionFactory sqlSessionFactory){ return new SqlSessionTemplate(sqlSessionFactory); } }
controller:
@RestController@Slf4jpublic class IndexController { @Autowired private UserService1 userService1; @RequestMapping("/multiDataSourceAdd") public int multiDataSourceAdd(){ return userService1.insert("spring boot2.0",23); } @Autowired private UserService2 userService2; @RequestMapping("/multiDataSourceAdd2") public int multiDataSourceAdd2(){ return userService2.insert("spring boot2.0",23); }}
访问http://127.0.0.1:8080/multiDataSourceAdd和http://127.0.0.1:8080/multiDataSourceAdd2查看两个数据库是否添加陈宫。
听说在Spring boot1.5的时候要指定1个主数据源通过添加@Primary可解决。
6.5.1 注意事项
在多数据源的情况下,使用@Transactional注解时,应该指定事务管理者@Transactional(transactionManager = "test...")
7 事务管理
7.1 SpringBoot整合事物管理
Springboot默认集成事物,只主要在方法上加上@Transactional即可
在多数据源的情况下如下使用,否则会报错,单数据源直接使用@Transactional,在6.5的基础上试试添加age为0 的记录。
//添加 @Transactional(transactionManager = "test1TransactionManager") public int insert(String name,Integer age){ int i=1/age; return userMapper.insert(name,age); }
7.2 spring boot解决分布式事务问题
7.2.1 怎么产生的?
以6.5的多数据源为例,如果添加数据的代码为以下代码:
@Servicepublic class UserService2 { @Autowired private UserMapper2 userMapper2; @Autowired private UserMapper1 userMapper1; //添加 @Transactional(transactionManager = "test2TransactionManager") public int insertUser1AndInsertUser2(String name,Integer age){ //第一个数据源 int t1 = userMapper2.insert(name, age); //第二个数据源 int t2 = userMapper1.insert(name, age); int i=1/age; return t1+t2; } }
此时访问http://127.0.0.1:8080/multiDataSourceAdd2?name=springboot2.2&age=0会两条都添加成功吗?还是仅有一条(哪一条)?还是两条都失败?
真实情况是int t2 = userMapper1.insert(name, age);添加成功,而另一条失败。原因很简单,一个事务对应一个数据源嘛,那么这种情况怎么解决呢?
7.2.2 spring boot使用jta+atomikos 解决分布式事务问题(传统方式)
添加依赖:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jta-atomikos</artifactId> </dependency>
配置文件.yml
mysql: datasource: # Mysql 1 test1: url: jdbc:mysql://localhost:3306/test?useSSL=false&useUnicode=true&characterEncoding=utf-8 username: root password: root minPoolSize: 3 maxPoolSize: 25 maxLifetime: 20000 borrowConnectionTimeout: 30 loginTimeout: 30 maintenanceInterval: 60 maxIdleTime: 60 # Mysql 2 test2: url: jdbc:mysql://localhost:3306/shiro?useSSL=false&useUnicode=true&characterEncoding=utf-8 username: root password: root minPoolSize: 3 maxPoolSize: 25 maxLifetime: 20000 borrowConnectionTimeout: 30 loginTimeout: 30 maintenanceInterval: 60 maxIdleTime: 60
读取配置文件类:
@Data@ConfigurationProperties(prefix = "mysql.datasource.test1")public class DB1Config { private String url; private String username; private String password; private int minPoolSize; private int maxPoolSize; private int maxLifetime; private int borrowConnectionTimeout; private int loginTimeout; private int maintenanceInterval; private int maxIdleTime; private String testQuery;} @Data@ConfigurationProperties(prefix = "mysql.datasource.test2")public class DB2Config { private String url; private String username; private String password; private int minPoolSize; private int maxPoolSize; private int maxLifetime; private int borrowConnectionTimeout; private int loginTimeout; private int maintenanceInterval; private int maxIdleTime; private String testQuery;}
数据源配置类,很明显,这里没有任何事物管理器,和6.5有很大区别。因为这里是将本地事务注册到了atomikos全局事务了,所以是没有的
@Configuration@MapperScan(basePackages = "com.itboy.spboot.test1.mapper",sqlSessionTemplateRef = "testSqlSessionTemplate" )public class MybatisConfig1 { // 配置数据源 @Bean(name = "testDataSource") public DataSource testDataSource(DB1Config testConfig) throws SQLException { MysqlXADataSource mysqlXaDataSource = new MysqlXADataSource(); mysqlXaDataSource.setUrl(testConfig.getUrl()); mysqlXaDataSource.setPinGlobalTxToPhysicalConnection(true); mysqlXaDataSource.setPassword(testConfig.getPassword()); mysqlXaDataSource.setUser(testConfig.getUsername()); mysqlXaDataSource.setPinGlobalTxToPhysicalConnection(true); AtomikosDataSourceBean xaDataSource = new AtomikosDataSourceBean(); xaDataSource.setXaDataSource(mysqlXaDataSource); xaDataSource.setUniqueResourceName("testDataSource"); xaDataSource.setMinPoolSize(testConfig.getMinPoolSize()); xaDataSource.setMaxPoolSize(testConfig.getMaxPoolSize()); xaDataSource.setMaxLifetime(testConfig.getMaxLifetime()); xaDataSource.setBorrowConnectionTimeout(testConfig.getBorrowConnectionTimeout()); xaDataSource.setLoginTimeout(testConfig.getLoginTimeout()); xaDataSource.setMaintenanceInterval(testConfig.getMaintenanceInterval()); xaDataSource.setMaxIdleTime(testConfig.getMaxIdleTime()); xaDataSource.setTestQuery(testConfig.getTestQuery()); return xaDataSource; } @Bean(name = "testSqlSessionFactory") public SqlSessionFactory testSqlSessionFactory(@Qualifier("testDataSource") DataSource dataSource) throws Exception { SqlSessionFactoryBean bean = new SqlSessionFactoryBean(); bean.setDataSource(dataSource); return bean.getObject(); } @Bean(name = "testSqlSessionTemplate") public SqlSessionTemplate testSqlSessionTemplate( @Qualifier("testSqlSessionFactory") SqlSessionFactory sqlSessionFactory) throws Exception { return new SqlSessionTemplate(sqlSessionFactory); } } @Configuration@MapperScan(basePackages = "com.itboy.spboot.test2.mapper",sqlSessionTemplateRef = "test2SqlSessionTemplate" )public class MybatisConfig2 { // 配置数据源 @Bean(name = "test2DataSource") public DataSource testDataSource(DB2Config testConfig) throws SQLException { MysqlXADataSource mysqlXaDataSource = new MysqlXADataSource(); mysqlXaDataSource.setUrl(testConfig.getUrl()); mysqlXaDataSource.setPinGlobalTxToPhysicalConnection(true); mysqlXaDataSource.setPassword(testConfig.getPassword()); mysqlXaDataSource.setUser(testConfig.getUsername()); mysqlXaDataSource.setPinGlobalTxToPhysicalConnection(true); //这里的意思就是将本地事务注册到Atomikos全局事务 AtomikosDataSourceBean xaDataSource = new AtomikosDataSourceBean(); xaDataSource.setXaDataSource(mysqlXaDataSource); xaDataSource.setUniqueResourceName("test2DataSource"); xaDataSource.setMinPoolSize(testConfig.getMinPoolSize()); xaDataSource.setMaxPoolSize(testConfig.getMaxPoolSize()); xaDataSource.setMaxLifetime(testConfig.getMaxLifetime()); xaDataSource.setBorrowConnectionTimeout(testConfig.getBorrowConnectionTimeout()); xaDataSource.setLoginTimeout(testConfig.getLoginTimeout()); xaDataSource.setMaintenanceInterval(testConfig.getMaintenanceInterval()); xaDataSource.setMaxIdleTime(testConfig.getMaxIdleTime()); xaDataSource.setTestQuery(testConfig.getTestQuery()); return xaDataSource; } @Bean(name = "test2SqlSessionFactory") public SqlSessionFactory testSqlSessionFactory(@Qualifier("test2DataSource") DataSource dataSource) throws Exception { SqlSessionFactoryBean bean = new SqlSessionFactoryBean(); bean.setDataSource(dataSource); return bean.getObject(); } @Bean(name = "test2SqlSessionTemplate") public SqlSessionTemplate testSqlSessionTemplate( @Qualifier("test2SqlSessionFactory") SqlSessionFactory sqlSessionFactory) throws Exception { return new SqlSessionTemplate(sqlSessionFactory); } }
通过在主类上添加@EnableConfigurationProperties(value = {DB1Config.class,DB2Config.class})注解运行,访问以下ip
http://127.0.0.1:8080/insertUser1AndInsertUser2?name=springboot2.2&age=0添加失败,表示分布式配置成功。这里注解掉6.5所配置的两个类,通过去掉
//@Configuration
//@MapperScan(basePackages = "com.itboy.spboot.test2.mapper",sqlSessionFactoryRef = "test2SqlSessionFactory")
两个注解即可。
8 缓存支持
8.1 注解配置与EhCache使用
添加依赖:
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-cache</artifactId></dependency>
新建ehcache.xml文件:
<?xml version="1.0" encoding="UTF-8"?><ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd" updateCheck="false"> <diskStore path="java.io.tmpdir/Tmp_EhCache" /> <!-- 默认配置 --> <defaultCache maxElementsInMemory="5000" eternal="false" timeToIdleSeconds="120" timeToLiveSeconds="120" memoryStoreEvictionPolicy="LRU" overflowToDisk="false" /> <cache name="baseCache" maxElementsInMemory="10000" maxElementsOnDisk="100000" /> <!-- name:缓存名称。 maxElementsInMemory:缓存最大个数。 eternal:对象是否永久有效,一但设置了,timeout将不起作用。 timeToIdleSeconds:设置对象在失效前的允许闲置时间(单位:秒)。仅当eternal=false对象不是永久有效时使用,可选属性,默认值是0,也就是可闲置时间无穷大。 timeToLiveSeconds:设置对象在失效前允许存活时间(单位:秒)。最大时间介于创建时间和失效时间之间。仅当eternal=false对象不是永久有效时使用,默认是0.,也就是对象存活时间无穷大。 overflowToDisk:当内存中对象数量达到maxElementsInMemory时,Ehcache将会对象写到磁盘中。 diskSpoolBufferSizeMB:这个参数设置DiskStore(磁盘缓存)的缓存区大小。默认是30MB。每个Cache都应该有自己的一个缓冲区。 maxElementsOnDisk:硬盘最大缓存个数。 diskPersistent:是否缓存虚拟机重启期数据 Whether the disk store persists between restarts of the Virtual Machine. The default value is false. diskExpiryThreadIntervalSeconds:磁盘失效线程运行时间间隔,默认是120秒。 memoryStoreEvictionPolicy:当达到maxElementsInMemory限制时,Ehcache将会根据指定的策略去清理内存。默认策略是LRU(最近最少使用)。你可以设置为FIFO(先进先出)或是LFU(较少使用)。 clearOnFlush:内存数量最大时是否清除。 --> </ehcache>
使用缓存:
@CacheConfig(cacheNames = "baseCache")public interface UserMapper { @Select("select * from users where name=#{name}") @Cacheable UserEntity findName(@Param("name") String name);}
清除缓存:
@Autowiredprivate CacheManager cacheManager;@RequestMapping("/remoKey")public void remoKey() { cacheManager.getCache("baseCache").clear();}
启动类@EnableCaching // 开启缓存注解运行
9 热部署
9.1 热部署原理
所谓的热部署:比如项目的热部署,就是在应用程序在不停止的情况下,实现新的部署
其实底层就是使用类加载器(classloader)重新读取字节码文件到jvm内存。
怎么实现呢?1、监听class文件是否有发生改变(版本号或者最近修改时间);2、当class文件发生改变时,使用classloader进行重新读取。
一般用于本地开发,提高开发效率,不需要重启服务器。
缺点:如果项目比较大,会非常卡--比较占内存
9.2 devtools
spring-boot-devtools 是一个为开发者服务的一个模块,其中最重要的功能就是自动应用代码更改到最新的App上面去。原理是在发现代码有更改之后,重新启动应用,但是速度比手动停止后再启动还要更快,更快指的不是节省出来的手工操作的时间。
其深层原理是使用了两个ClassLoader,一个Classloader加载那些不会改变的类(第三方Jar包),另一个ClassLoader加载会更改的类,称为 restart ClassLoader ,这样在有代码更改的时候,原来的restart ClassLoader 被丢弃,重新创建一个restart ClassLoader,由于需要加载的类相比较少,所以实现了较快的重启时间(5秒以内)
当修改controller里面的代码,比如新增一个requestMapping时,他就会将其注册到MVC容器中,这样就相当于重启了嘛。
添加依赖:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-devtools</artifactId> <optional>true</optional> <scope>true</scope> </dependency>
1. devtools会监听classpath下的文件变动,并且会立即重启应用(发生在保存时机),注意:因为其采用的虚拟机机制,该项重启是很快的。
2. devtools可以实现页面热部署(即页面修改后会立即生效,这个可以直接在application.properties文件中配置spring.thymeleaf.cache=false来实现(这里注意不同的模板配置不一样)
10 2.0新特性
10.1 以Java8为基准
Spring Boot 2.0 要求Java 版本必须8以上, Java 6 和 7 不再支持。
10.2 内嵌容器包结构调整
为了支持reactive使用场景,内嵌的容器包结构被重构了的幅度有点大。EmbeddedServletContainer被重命名为WebServer,并且org.springframework.boot.context.embedded 包被重定向到了org.springframework.boot.web.embedded包下。举个例子,如果你要使用TomcatEmbeddedServletContainerFactory回调接口来自定义内嵌Tomcat容器,你现在应该使用TomcatServletWebServerFactory。
10.3 Servlet-specific 的server properties调整
Old property | New property |
server.context-parameters.* | server.servlet.context-parameters.* |
server.context-path | server.servlet.context-path |
server.jsp.class-name | server.servlet.jsp.class-name |
server.jsp.init-parameters.* | server.servlet.jsp.init-parameters.* |
server.jsp.registered | server.servlet.jsp.registered |
server.servlet-path | server.servlet.path |
server不再是只有servlet了,还有其他的要加入。
10.4 Actuator 默认映射
Actuator的端点(endpoint)现在默认映射到/application,比如,/info 端点现在就是在/application/info。但你可以使用management.context-path来覆盖此默认值。
10.5 Spring Loaded不再支持
由于Spring Loaded项目已被移到了attic了,所以不再支持Spring Loaded了。现在建议你去使用Devtools。Spring Loaded不再支持了。
10.6 支持Quartz Scheduler
Spring Boot 2 针对Quartz调度器提供了支持。你可以加入spring-boot-starter-quartz starter来启用。而且支持基于内存和基于jdbc两种存储
10.7 支持Spring WebFlux
WebFlux 模块的名称是 spring-webflux,名称中的 Flux 来源于 Reactor 中的类 Flux。该模块中包含了对反应式 HTTP、服务器推送事件和 WebSocket 的客户端和服务器端的支持。对于开发人员来说,比较重要的是服务器端的开发。在服务器端,WebFlux 支持两种不同的编程模型:第一种是 Spring MVC 中使用的基于 Java 注解的方式;第二种是基于 Java 8 的 lambda 表达式的函数式编程模型。这两种编程模型只是在代码编写方式上存在不同。它们运行在同样的反应式底层架构之上,因此在运行时是相同的。WebFlux 需要底层提供运行时的支持,WebFlux 可以运行在支持 Servlet 3.1 非阻塞 IO API 的 Servlet 容器上,或是其他异步运行时环境,如 Netty 和 Undertow。
10.8 版本要求
Jetty
要求Jetty最低版本为9.4。
Tomcat
要求Tomcat最低版本为8.5。
Hibernate
要求Hibernate最低版本为5.2。
Gradle
要求Gradle最低版本为3.4。
SendGrid
SendGrid最低支持版本是3.2。为了支持这次升级,username和password已经被干掉了。因为API key现在是唯一支持的认证方式。
11 Spring boot监控中心
11.1 Springboot监控中心概述
Spring boot监控中心:针对微服务服务器监控,服务器内存变化(堆内存、线程、日志管理等)、检测服务配置连接地址是否可用(比如一些mysql、redis的配置,当懒加载的情况,就需要模拟请求去检测一下)、统计现在有多少个bean(Spring容器中的bean实例)、统计SpringMVC@RequestMapping(http接口)。
Actuator监控:没有界面、返回json格式
AdminUI:底层使用Actuator监控应用 ,实现可视化页面
应用场景:生产环境
为什么要有这个东西?
Actuator是Springboot的一个附加功能,可帮助你在应用程序生产环境时监视和管理应用程序。可以使用HTTP的各种请求来监管,审计,收集应用的运行情况,对于微服务管理十分有意义。缺点:没有可视化界面。
11.2 Actuator
默认情况下开启只开启三个接口,Spring boot2.0之后接口发生了变化,需要加一个/actuator,不过可以通过配置文件修改。
添加依赖:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency>
配置文件:
#开启所有接口,默认情况下只有info、health、/三个management: endpoints: web: exposure: include: "*"
启动应用后可以看到增加了很多/actuator/..这样的接口。
测试一下检测服务配置连接是否可用?新增一个mysql连接配置和依赖后,如果密码错误默认是懒加载形式,程序启动正常。但是通过访问http://localhost:8080/actuator/health这个接口后发现程序抛出异常返回{status:"down"},说明模拟请求发送一个请求到mysql,而密码错误所以抛出异常,状态为down否则为up。
常用:
路径 | 作用 |
/actuator/beans | 显示应用程序中所有Spring bean的完整列表。 |
/actuator/configprops | 显示所有配置信息。 |
/actuator/env | 陈列所有的环境变量。 |
/actuator/mappings | 显示所有@RequestMapping的url整理列表。 |
/actuator/health | 显示应用程序运行状况信息 up表示成功 down失败 |
/actuator/info | 查看自定义应用信息,配置文件中以info开头的 |
11.3 AdminUI
11.3.1 原理
Admin-UI基于actuator实现能够返回界面展示监控信息。
1、服务1和服务2两个服务内部集成自己的Actualtor监控应用,比如开发哪些接口啊等等,之后将这些接口注册到AdminUI平台上进行展示。
2、或者 说对于服务1和服务2两个服务(类似于client),将他们的监控中心管理存放在AdminUI(类似于server)平台上。
这样当AdminUI平台要展示服务1或者2的时候,服务1/2通过Actuator发送一个json串(Actuator返回的就是json)给AdminUI,AdminUI解析这个json串然后再页面上进行展示。
11.3.2 搭建一个AdminUI server
添加依赖:
<dependency> <groupId>de.codecentric</groupId> <artifactId>spring-boot-admin-starter-server</artifactId> <version>2.0.0</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-webflux</artifactId> </dependency> <!-- Spring Boot Actuator对外暴露应用的监控信息,Jolokia提供使用HTTP接口获取JSON格式 的数据 --> <dependency> <groupId>org.jolokia</groupId> <artifactId>jolokia-core</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency> <dependency> <groupId>com.googlecode.json-simple</groupId> <artifactId>json-simple</artifactId> <version>1.1</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency>
配置文件:
spring: application: name: spring-boot-admin-server
启动类:
@Configuration@EnableAutoConfiguration@EnableAdminServerpublic class AdminUiServerApplication { public static void main(String[] args) { SpringApplication.run(AdminUiServerApplication.class, args); }}
访问http://localhost:8080检查是否成功。
11.3.3 搭建一个AdminUI client
添加依赖:
<dependency> <groupId>de.codecentric</groupId> <artifactId>spring-boot-admin-starter-client</artifactId> <version>2.0.0</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-actuator</artifactId> </dependency> <dependency> <groupId>org.jolokia</groupId> <artifactId>jolokia-core</artifactId> </dependency> <dependency> <groupId>com.googlecode.json-simple</groupId> <artifactId>json-simple</artifactId> <version>1.1</version> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency>
配置文件:
spring: boot: admin: client: url: http://localhost:8080server: port: 8081 management: endpoints: web: exposure: include: "*" endpoint: health: show-details: ALWAYS
启动类:
@SpringBootApplicationpublic class AdminUiClientApplication { public static void main(String[] args) { SpringApplication.run(AdminUiClientApplication.class, args); }}
访问http://localhost:8081/actuator/health是否成功以及刷新http://localhost:8080后是否注册成功。