SpringBoot
1 SpringBoot概述敏捷开发(整合框架,直接加入依赖就好,自动配置);无须Tomcat(Java应用程序方式运行,实际jar包),内置Tomcat减少xml配置(甚至没有),配置文件properties/yml注解形式SpringBoot 是一个快速开发的框架,能够快速的整合第三方框架,简化XML配置,全部采用注解形式,内置Tomcat容器,帮助开发者能够实现快速开发,...
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
@EnableAutoConfiguration
public 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")
@EnableAutoConfiguration
public class App {
public static void main(String[] args) {
SpringApplication.run(App.class, args);
}
}
2.3 springboot启动方式3
@SpringBootApplication 被 @Configuration、@EnableAutoConfiguration、@ComponentScan 注解所修饰,换言之 Springboot 提供了统一的注解来替代以上三个注解
扫包范围:在启动类上加上@SpringBootApplication注解,当前包下或者子包下所有的类都可以扫到。
@SpringBootApplication
public 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=false
spring.freemarker.cache=true
spring.freemarker.check-template-location=true
spring.freemarker.charset=UTF-8
spring.freemarker.content-type=text/html
spring.freemarker.expose-request-attributes=false
spring.freemarker.expose-session-attributes=false
spring.freemarker.expose-spring-macro-helpers=false
#spring.freemarker.prefix=
#spring.freemarker.request-context-attribute=
#spring.freemarker.settings.*=
spring.freemarker.suffix=.ftl
spring.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
@Controller
public 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,DEBUG
log4j.rootLogger=info,error,CONSOLE,DEBUG
log4j.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=info
log4j.appender.info=org.apache.log4j.DailyRollingFileAppender
log4j.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-dd
log4j.appender.info.Threshold = info
log4j.appender.info.append=true
#log4j.appender.info.File=/home/admin/pms-api-services/logs/info/api_services_info
log4j.appender.info.File=/Users/dddd/Documents/testspace/pms-api-services/logs/info/api_services_info
log4j.logger.error=error
log4j.appender.error=org.apache.log4j.DailyRollingFileAppender
log4j.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-dd
log4j.appender.error.Threshold = error
log4j.appender.error.append=true
#log4j.appender.error.File=/home/admin/pms-api-services/logs/error/api_services_error
log4j.appender.error.File=/Users/dddd/Documents/testspace/pms-api-services/logs/error/api_services_error
log4j.logger.DEBUG=DEBUG
log4j.appender.DEBUG=org.apache.log4j.DailyRollingFileAppender
log4j.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-dd
log4j.appender.DEBUG.Threshold = DEBUG
log4j.appender.DEBUG.append=true
#log4j.appender.DEBUG.File=/home/admin/pms-api-services/logs/debug/api_services_debug
log4j.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
@Component
public 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
@Data
public 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
@Slf4j
public 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
@Slf4j
public 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
@Slf4j
public 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=8080
server.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
@Component
public 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;
}
}
注册拦截器:
@Configuration
public 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
@Slf4j
public class IndexController {
@Autowired
private IndexService indexService;
@RequestMapping("/createUser")
public String createUser(){
int xiaoming = indexService.createUser("xiaoming", 23);
return "success:"+xiaoming;
}
}
IndexService.java
@Service
@Slf4j
public 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>
实体类:
@Data
public class User {
private int id;
private String name;
private Integer age;
}
mapper类:
@Mapper
public 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类:
@Service
public 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
@Slf4j
public 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: DEBUG
pagehelper:
helper-dialect: mysql
reasonable: true
support-methods-arguments: true
params: count=countSql
page-size-zero: true
mapper:
@Mapper
public interface UserMapper {
//列表查询
@Select("SELECT * FROM USERS")
List<User> findUserList();
}
service:
@Service
public 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
@Slf4j
public 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
@Slf4j
public 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都有
@Data
public 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都有,这里仅展示一个
@Service
public 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
@Slf4j
public 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的多数据源为例,如果添加数据的代码为以下代码:
@Service
public 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);
}
清除缓存:
@Autowired
private 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
@EnableAdminServer
public 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:8080
server:
port: 8081
management:
endpoints:
web:
exposure:
include: "*"
endpoint:
health:
show-details: ALWAYS
启动类:
@SpringBootApplication
public class AdminUiClientApplication {
public static void main(String[] args) {
SpringApplication.run(AdminUiClientApplication.class, args);
}
}
访问http://localhost:8081/actuator/health是否成功以及刷新http://localhost:8080后是否注册成功。
更多推荐
所有评论(0)