目录
spring发展史
Springboot简介
Springboot使用
环境信息
idea中搭建一个springboot工程
springboot注解说明
热部署
springboot启动时加载配置文件
springboot @Mapper使用
springboot mybatis使用别名
springboot mybatis设置map返回时 如果为空返回null值
springboot 设置过滤器
springboot 设置拦截器
springboot servlet上下文监听器
springboot 定时任务
springboot 数据库连接池配置
默认数据库连接池:hikari 号称“史上最快连接池”
Druid数据库连接池(阿里自研可以很方便的监控性能)
springboot lombok
springboot 日志配置
spring发展史
第一阶段:xml配置
在Spring 1.x时代,使用Spring开发满眼都是xml配置的Bean,随着项目的扩大,我们需要把xml配置文件放到不同的配置文件里,那时需要频繁的在开发的类和配置文件之间进行切换
第二阶段:注解配置
在Spring 2.x 时代,随着JDK1.5带来的注解支持,Spring提供了声明Bean的注解(例如@Component、@Service),大大减少了配置量。主要使用的方式是应用的基本配置(如数据库配置)用xml,业务配置用注解
第三阶段:java配置
Spring 3.0 引入了基于 Java 的配置能力,这是一种类型安全的可重构配置方式,可以代替 XML。我们目前刚好处于这个时代,Spring4.x和Spring Boot都推荐使用Java配置。
Springboot简介
简介:SpringBoot是由Pivotal团队在2013年开始研发、2014年4月发布第一个版本的全新开源的轻量级框架。它基于Spring4.0设计,不仅继承了Spring框架原有的优秀特性,而且还通过简化配置来进一步简化了Spring应用的整个搭建和开发过程。另外SpringBoot通过集成大量的框架使得依赖包的版本冲突,以及引用的不稳定性等问题得到了很好的解决。
特点:
(1)可以创建独立的Spring应用程序,并且基于其Maven或Gradle插件,可以创建可执行的JARs和WARs;
(2)内嵌Tomcat或Jetty等Servlet容器;
(3)提供自动配置的“starter”项目对象模型(POMS)以简化Maven配置;
(4)尽可能自动配置Spring容器;
(5)提供准备好的特性,如指标、健康检查和外部化配置;
(6)绝对没有代码生成,不需要XML配置。
策略:
(1)开箱即用,Outofbox,是指在开发过程中,通过在MAVEN项目的pom文件中添加相关依赖包,然后使用对应注解来代替繁琐的XML配置文件以管理对象的生命周期。这个特点使得开发人员摆脱了复杂的配置工作以及依赖的管理工作,更加专注于业务逻辑。
(2)约定优于配置,Convention over configuration,是一种由SpringBoot本身来配置目标结构,由开发者在结构中添加信息的软件设计范式。这一特点虽降低了部分灵活性,增加了BUG定位的复杂性,但减少了开发人员需要做出决定的数量,同时减少了大量的XML配置,并且可以将代码编译、测试和打包等工作自动化。
Springboot使用
环境信息
数据库:MySQL
IDE:idea
Spring-Boot:2.2.6
Maven: 3.3.9
idea中搭建一个springboot工程
传送门:
springboot注解说明
引导类中使用
//注解表示当前类是springboot的应用
@SpringBootApplication
//设置包扫描路径
@ComponentScan(basePackages = "com.anran.springboot")
配置类中使用
// 配置类说明
@Configuration
Bean类使用
// 接口配置(@Controller和@ResponseBody)
@RestController
// 接口配置
@Controller
// 路径对外别名,请求方式以及请求参数类型
@RequestMapping("/testController")
// 返回信息说明
@ResponseBody
// bean的声明
@Service
// dao bean的声明
@Mapper
// 依赖注入
@Autowired
// lombok的注解,用于自动生成对象属性的get和set方法
@Data
过滤器注解
// 过滤器描述
@WebFilter(filterName = "myFilter", urlPatterns = "/*") (public class MyFilter implements Filter)
servlt上下文监听器
@WebListener (public class MyListener implements ServletContextListener)
定时任务
// 开启定时任务
@EnableScheduling
// 定时逻辑
@Scheduled(cron/fixedRate/fixedDelay/initialDelay)
声明bean
@Bean:声明在方法上,将方法的返回值加入Bean容器,代替标签
@PropertySource:指定外部属性文件
热部署
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-devtools</artifactId>
</dependency>
springboot启动时加载配置文件
项目在启动时会查找config/classpath目录下的application文件(后缀包含properties、xml、yml、yaml)
以yml配置说明为例
application.yml(设置环境,用于多环境部署)
spring:
profiles:
active: dev
application-dev.yml(开发环境配置)
server:
port: 8082
spring:
datasource:
username: root
password: jkgjghjg9
url: jdbc:mysql://localhost:3306/project_manage?useUnicode=true&characterEncoding=utf-8&useSSL=true&serverTimezone=UTC
driver-class-name: com.mysql.jdbc.Driver
mybatis:
mapper-locations: classpath:mapper/*Mapper.xml
type-aliases-package: com.anran.example.model
logging:
file: /usr/log/example.log
level:
root: INFO
com.anran.projectmanage.mapper: DEBUG
springboot @Mapper使用
方式1:
package com.anran.example.mapper;
import com.anran.example.model.TestModel;
import org.apache.ibatis.annotations.*;
@Mapper
public interface MyTestMapper {
// 查找信息
@Select("select id, date_format(now(), '%Y%m%d%H%i%s') as content from test where id = #{id}")
TestModel get(int id);
// 插入信息
@Insert({ "insert into test(content) values(#{content})" })
// 返回自增主键
@Options(useGeneratedKeys = true, keyProperty = "id")
void insert(TestModel model);
// 插入信息
@Update({ "update test set content = #{content} where id = #{id}" })
int update(TestModel model);
// 删除信息
@Update({ "delete from test where id = #{id}" })
int delete(int id);
}
方式2;
package com.anran.example.mapper;
import com.anran.example.model.TestModel;
import org.apache.ibatis.annotations.Mapper;
@Mapper
public interface TestMapper {
TestModel get(int id);
}
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.anran.example.mapper.TestMapper">
<select id="get" parameterType="int" resultType="com.anran.example.model.TestModel">
select
id, content
from
test
where
id = #{id};
</select>
</mapper>
mybatis:
mapper-locations: classpath:mapper/*Mapper.xml
type-aliases-package: com.anran.example.model
springboot mybatis使用别名
mybatis:
# 设置mapper映射类
mapper-locations: classpath:mapper/*Mapper.xml
# 设置model模型别名
type-aliases-package: com.anran.example.model
mapper.xml中使用
<select id="get" parameterType="int" resultType="TestModel">
select id, content from test where id = #{id};
</select>
springboot mybatis设置map返回时 如果为空返回null值
mybatis:
# 设置mapper映射类
mapper-locations: classpath:mapper/*Mapper.xml
# 设置model模型别名
type-aliases-package: com.anran.example.model
# 设置map结果时如果是null也要返回key
configuration:
call-setters-on-nulls: true
springboot 设置过滤器
方式1:@WebFilter
package com.anran.example.filter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
@WebFilter(filterName = "myFilter", urlPatterns = "/*")
public class MyFilter implements Filter {
private static final Logger log = LoggerFactory.getLogger(MyFilter.class);
@Override
public void init(FilterConfig filterConfig) throws ServletException {
log.info("MyFilter init");
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
log.info("MyFilter doFilter");
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
}
}
方式2:@Configuration和@Bean
配置文件:MyConfig.java
package com.anran.example.filter;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class MyConfig {
@Bean
public FilterRegistrationBean registMyFilter1() {
FilterRegistrationBean registration = new FilterRegistrationBean(new MyFilter1());
registration.addUrlPatterns("/*"); // 过滤url
registration.addInitParameter("paramName", "paramValue"); // 初始化参数
registration.setName("MyFilter1"); // 过滤器名称
registration.setOrder(1); // 过滤器顺序
return registration;
}
@Bean
public FilterRegistrationBean registMyFilter2() {
FilterRegistrationBean registration = new FilterRegistrationBean(new MyFilter2());
registration.addUrlPatterns("/*"); //
registration.addInitParameter("paramName", "paramValue"); //
registration.setName("MyFilter2");
registration.setOrder(2);
return registration;
}
}
过滤器1:
package com.anran.example.filter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
public class MyFilter1 implements Filter {
private static final Logger log = LoggerFactory.getLogger(MyFilter1.class);
@Override
public void init(FilterConfig filterConfig) throws ServletException {
log.info("MyFilter1 init");
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
log.info("MyFilter1 doFilter");
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
}
}
过滤器2:
package com.anran.example.filter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import java.io.IOException;
public class MyFilter2 implements Filter {
private static final Logger log = LoggerFactory.getLogger(MyFilter2.class);
@Override
public void init(FilterConfig filterConfig) throws ServletException {
log.info("MyFilter2 init");
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
log.info("MyFilter2 doFilter");
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
}
}
springboot 设置拦截器
package com.anran.example.filter;
import com.anran.example.Interceptor.MyInterceptor;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
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.ViewControllerRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
@Configuration
public class MyConfig implements WebMvcConfigurer {
@Override
public void addViewControllers(ViewControllerRegistry registry) {
registry.addViewController("/zxc/foo").setViewName("foo");
}
// 拦截器
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new MyInterceptor())
.addPathPatterns("/*");
}
}
package com.anran.example.Interceptor;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class MyInterceptor implements HandlerInterceptor {
private static final Logger log = LoggerFactory.getLogger(MyInterceptor.class);
// 接口请求之前
@Override
public boolean preHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o) throws Exception {
log.info("preHandle被调用");
// true请求继续,false请求直接结束(参数校验,用户登录)
return true;
}
// 接口请求之后,返回之前
@Override
public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {
System.out.println("postHandle被调用");
}
// 返回之后
@Override
public void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {
System.out.println("afterCompletion被调用");
}
}
springboot servlet上下文监听器
package com.anran.example.listener;
import com.anran.example.Interceptor.MyInterceptor;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener;
@WebListener
public class MyListener implements ServletContextListener {
private static final Logger log = LoggerFactory.getLogger(MyListener.class);
@Override
public void contextDestroyed(ServletContextEvent sce) {
log.info("MyListener destory");
}
@Override
public void contextInitialized(ServletContextEvent sce) {
log.info("MyListener init");
}
}
springboot 定时任务
传送门:
springboot 数据库连接池配置
默认数据库连接池:hikari 号称“史上最快连接池”
application.yml
spring:
datasource:
username: root
password: axxxxxxxxxxxxx9
url: jdbc:mysql://localhost:3306/project_manage?useUnicode=true&characterEncoding=utf-8&useSSL=true&serverTimezone=UTC
driver-class-name: com.mysql.jdbc.Driver
# Hikari 连接池配置
hikari:
# 空闲连接存活最大时间,默认600000(10分钟)
idle-timeout: 180000
# 最小空闲连接数量
minimum-idle: 5
# 连接池最大连接数,默认是10
maximum-pool-size: 10
# 此属性控制从池返回的连接的默认自动提交行为,默认值:true
auto-commit: true
# 连接池名称
pool-name: MyHikariCP
# 此属性控制池中连接的最长生命周期,值0表示无限生命周期,默认1800000即30分钟
max-lifetime: 1800000
# 数据库连接超时时间,默认30秒,即30000
connection-timeout: 30000
# 测试验证查询
connection-test-query: SELECT 1
Druid数据库连接池(阿里自研可以很方便的监控性能)
pom.xml
<!-- Druid连接池 --> <dependency> <groupId>com.alibaba</groupId> <artifactId>druid-spring-boot-starter</artifactId> <version>1.1.10</version> </dependency>
application.yml
spring:
# 配置数据库信息
datasource:
druid:
# 数据源配置
username: root
password: ajl930919
url: jdbc:mysql://localhost:3306/project_manage?useUnicode=true&characterEncoding=utf-8&useSSL=true&serverTimezone=UTC
driver-class-name: com.mysql.cj.jdbc.Driver
# 初始化 最小 最大
initial-size: 5
min-idle: 5
max-active: 20
# 配置获取连接等待超时的时间
max-wait: 60000
# 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒
time-between-eviction-runs-millis: 60000
# 配置一个连接在池中最小生存的时间,单位是毫秒
min-evictable-idle-time-millis: 300000
validation-query: SELECT 'x'
test-while-idle: true
test-on-borrow: false
test-on-return: false
# 打开PSCache,并且指定每个连接上PSCache的大小
poolPreparedStatements: true
maxPoolPreparedStatementPerConnectionSize: 20
# 配置多个英文逗号分隔
filters: stat,wall
# WebStatFilter配置,说明请参考Druid Wiki,配置_配置WebStatFilter
# 是否启用StatFilter默认值true
web-stat-filter:
enabled: true
url-pattern: /*
exclusions: "*.js,*.gif,*.jpg,*.png,*.css,*.ico,/druid/*"
session-stat-enable: false
session-stat-max-count: 1000
principal-cookie-name: admin
principal-session-name: admin
profile-enable: true
# 根据配置中的url-pattern来访问内置监控页面,如果是上面的配置,内置监控页面的首页是/druid/index.html
# http://loacalhsot:8081/druid
stat-view-servlet:
url-pattern: /druid/* # 监控页面访问路径
# 允许清空统计数据
reset-enable: true
login-username: admin
login-password: 123456
# StatViewSerlvet展示出来的监控信息比较敏感,是系统运行的内部情况,如果你需要做访问控制,可以配置allow和deny这两个参数
# deny优先于allow,如果在deny列表中,就算在allow列表中,也会被拒绝。如果allow没有配置或者为空,则允许所有访问
# 配置的格式
# <IP>
# 或者<IP>/<SUB_NET_MASK_size>其中128.242.127.1/24
# 24表示,前面24位是子网掩码,比对的时候,前面24位相同就匹配,不支持IPV6。
allow:
deny:
DruidStatController.java
package com.anran.example.controller;
import com.alibaba.druid.stat.DruidStatManagerFacade;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class DruidStatController {
@GetMapping("/durid/stat")
public Object druidStat(){
// DruidStatManagerFacade#getDataSourceStatDataList 该方法可以获取所有数据源的监控数据,除此之外 DruidStatManagerFacade 还提供了一些其他方法,你可以按需选择使用。
return DruidStatManagerFacade.getInstance().getDataSourceStatDataList();
}
}
浏览器输入:http://localhost:8082/druid/login.html 账号:admin 密码:123456
ps:druid使用1.1.10以上版本出现页面无法访问,暂未解决
springboot lombok
idea配置lombok传送门:
springboot 日志配置
Spring Boot 默认的日志框架 logback,spring-boot-starter其中包含了 spring-boot-starter-logging的依赖。
application.yml
logging:
# 日志级别
level:
# 没有设置的统一级别
root: INFO
# 对应包下面的日志级别
com.anran.projectmanage.mapper: DEBUG
file:
# 日志名称(path和name不能同时使用)
name: /usr/log/example.log
# 日志大小
max-size: 1MB
# 历史日志备份文件时间(天)
max-history: 7
pattern:
# 配置控制台日志输出格式(时间+级别+线程+包名+方法+行数+输出信息)
console: '%d %-5level [%thread] %logger:%method:%line %msg %n'
# 配置文件日志输出格式
file: '%d %-5level [%thread] %logger:%method:%line %msg %n'
使用
方式1:lombok
@Slf4j
log.info("log finish id : ", id);
方式2:自己声明
private static final Logger log = LoggerFactory.getLogger(TestController.class);
log.info("log finish id : ", id);