SpringBoot最全笔记,企业最核心的技术你确定不来看看?

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介:  SpringBoot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置。通过这种方式,Spring Boot致力于在蓬勃发展的快速应用开发领域(rapid application development)成为领导者。    SpringBoot能够快发开发的原因是因为配置文件从xml转移到了java文件中,减少了配置文件的书写。

1.JPG


一、SpringBoot入门


1.1、SpringBoot介绍


   SpringBoot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置。通过这种方式,Spring Boot致力于在蓬勃发展的快速应用开发领域(rapid application development)成为领导者。     SpringBoot能够快发开发的原因是因为配置文件从xml转移到了java文件中,减少了配置文件的书写。


1.2、JavaConfig


   JavaConfig的技术是用于替代目前繁琐的xml文件的配置,他最为重要的替代方式就是使用了大量的注解。


1.2.1、项目准备


<properties>
        <spring.version>5.0.8.RELEASE</spring.version>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <maven.compiler.source>11</maven.compiler.source>
        <maven.compiler.target>11</maven.compiler.target>
</properties>
<dependencies>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-test</artifactId>
        <version>${spring.version}</version>
    </dependency>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <version>1.16.22</version>
        <scope>provided</scope>
    </dependency>
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.12</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context-support</artifactId>
        <version>${spring.version}</version>
    </dependency>
</dependencies>
复制代码

1.2.2、配置类替代配置文件之控制反转

1.2.2.1、创建Bean

public class OneBean {
}
复制代码


1.2.2.2、创建配置类


// 配置类注解,贴上表明这个类是一个配置类
@Configuration
public class AppConfig {
    // Bean实例注解,贴有该注解的方法为实例方法,在功能上等价于:<bean name="someBean" class="cn.linstudy.onfig.OmeBean" ></bean>
    @Bean
    public OneBean oneBean(){
        // 注意:实例方法的返回对象会交由Spring容器管理起来
        return new SomeBean();
    }
}
复制代码


1.2.2.3、测试


@RunWith(SpringRunner.class)
// 引用配置类的注解
@ContextConfiguration(classes = AppConfig.class) 
public class App {
    @Autowired
    private ApplicationContext ctx;
    @Test
    public void testApp(){
        OmeBean omeBean = ctx.getBean("omeBean", SomeBean.class);
        System.out.println(omeBean);
    }
}
复制代码


1.2.3、配置类替代配置文件之组件扫描


1.2.3.1、Bean组件扫描


   学习spring框架时, Spring有4个版型标签(代表的含义相同,仅仅只是为了标注表示的对象不同)。当spring容器扫描器扫描到贴有版型标签的类,会自动创建这些类的实例对象,并交给容器管理。


@Controller  //标记控制层
public class EmployeeController{
}
@Service    //标记服务层
public class EmployeeServiceImpl{
}
@Repository  //标记持久层
public class EmployeeDAOImpl{
}
@Component   //其他类
public class EmployeeListener{
}
复制代码

   我们除了使用上述注解,还需要在xml文件中进行配置。

<context:component-scan base-package="指定扫描的包路径"></context:component-scan>
复制代码

   换成了JavaConfig,我们需要贴注解来告诉Spring需要扫描这个类并且创建对象。

@Component  //版型标签
public class OmeBean {
}
复制代码

 

spring组件扫描注解, 扫描basePackages属性指定包及其子包下所有的贴有版型标签类,并创建对象交给Spring容器管理。如果不指定basePackages属性,表示扫描当前类所有包及其子包。


@Configuration
//组件扫描标签
@ComponentScan(basePackages ="cn.linstudy.config") 
public class AppConfig {
}
复制代码


1.2.4、@Bean注解详解


1.2.4.1、创建一个OneBean类


@Setter
@Getter
public class OneBean {
    public OneBean() {
        System.out.println("OneBean被创建");
    }
    public void init() {
        System.out.println("OneBean被初始化");
    }
    public void destroy() {
        System.out.println("OneBean被销毁");
    }
}
复制代码


1.2.4.2、书写配置类


@Scope("singleton")
@Bean(value = "ob",initMethod = "init", destroyMethod = "destroy")
public OomeBean oomeBean(){
    return new OomeBean();
}
复制代码


1.2.4.3、总结


  1. bean标签中的name属性 = @Bean注解中的name/value属性。
  2. bena标签中的id属性 = 实例方法的方法名。
  3. bean标签中的init-method = @Bean注解中的initMethod属性。
  4. bean标签中destroy-method属性 = @Bean注解中destroyMethod属性。
  5. bean标签中scope属性 = 实例方法中的@Scope注解


1.2.5、配置类替代配置文件之依赖注入


1.2.5.1、创建新类


public class TwoBean {
}
复制代码
@Setter
@Getter
public class OneBean {
    private Two twoBean;
}
复制代码


1.2.5.1、实现方式一


   我们可以直接调用实例方法来实现依赖注入。


   JavaConfig实现传统依赖注意需要注意:


  1. omeBean对象可以从容器中获取。
  2. twoBean可以从容器中获取。
  3. omeBean对象通过getTwoBean()方法获得的bean应该从容器获取的twoBean对象相等。


@Configuration
public class AppConfig {
    @Bean
    public OmeBean omeBean(){
        OmeBean omeBean = new OmeBean();
        OmeBean.setTwoBean(twoBean());
        return someBean;
    }
    @Bean
    public TwoBean twoBean(){
        return new TwoBean();
    }
}
复制代码

 

 需要注意的点:

  1. 多次调用twoBean()方法, spring容器只会执行一次twoBean对象的构建,原因:twoBean()方法被spring容器代理了,每次调用前都会执行容器bean检查,当发现容器中已经有了,直接从容器中拿。如果没有,则执行方法,并将返回值放置到容器中。


1.2.5.2、实现方式二


   方式二是可以通过注入实例对象的方式来实现依赖注入。


@Configuration
public class AppConfig {
    @Bean
    public OneBean omeBean(TwoBean twoBean){
        OneBean omeBean = new OneBean();
        omeBean.setTwoBean(twoBean);
        return omeBean;
    }
    @Bean
    public TwoBean twoBean(){
        return new TwoBean();
    }
}
复制代码


1.2.6、配置文件互相导入


1.2.6.1、@import


   配置类导入注解,贴在配置类上,作用等价于:<import resource="xxx配置.xml"></import>标签该标签用于配置类与配置类间的导入。


1.2.6.2、@ImportResource


   配置文件导入注解,贴在配置类上,作用等价于:<import resource="xxx配置.xml"></import>标签该标签用于配置类与配置文件间的导入。


1.2.7、配置文件的加载与取值


1.2.7.1、加载


   我们可以使用@PropertySource这个注解来进行资源配置文件加载注解,贴在配置类上,用于将properties类型文件加载到spring容器中。


<context:property-placeholder location="classpath:xxx.perperties"/>
复制代码


1.2.7.2、取值


   当加载了一个配置文件的时候,如果我们需要取值,可以使用@Value注解从properties配置中获取配置的数据。


1.2.7.2.1、创建db.properties


jdbc.driverClassName=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://192.168.126.129:3306/car_crm
jdbc.username=root
jdbc.password=123456
复制代码

1.2.7.2.2、定义一个类


@Setter
@Getter
@ToString
public class MyDataSource {
    private String driverClassName;
    private String url;
    private String username;
    private String password;
}
复制代码

1.2.7.2.3、接收值


@PropertySource("classpath:db.properties")
@Configuration
public class AppConfig {
    @Value("${jdbc.driverClassName}")
    private String driverClassName;
    @Value("${jdbc.url}")
    private String url;
    @Value("${jdbc.username}")
    private String username;
    @Value("${jdbc.password}")
    private String password;
    @Bean
    public MyDataSource myDataSource(){
        MyDataSource source = new MyDataSource();
        source.setDriverClassName(driverClassName);
        source.setUrl(url);
        source.setUsername(username);
        source.setPassword(password);
        return source;
    }
}
复制代码


1.3、SpringBoot自动装配原理


1.3.1、SpringBoot的核心注解


   SpringBoot的核心是@SpringBootApplication这个注解。


@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = {
        @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
        @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {
复制代码


1.3.2、SpringBoot自动装配原理。


1.3.2.1、综述


   SpringBoot启动会加载大量的自动配置类。@SpringBootApplication包含以下三个注解:


  1. @SpringBootConfiguration:我们点进去以后可以发现底层是Configuration注解,说白了就是支持JavaConfig的方式来进行配置(使用Configuration配置类等同于XML文件)。
  2. @EnableAutoConfiguration:开启自动配置功能。
  3. @ComponentScan:这个就是扫描注解,默认是扫描当前类下的package。将@Controller/@Service/@Component/@Repository等注解加载到IOC容器中。

   

我们进入到@EnableAutoConfiguration,就会发现他导入了@implort注解导入了AutoConfigurationImportSelector配置类,该类有一个getCandidateConfiguration方法获取候选的配置方法,可以读取依赖中META-INF/spring.factories中各种候选的配置类,根据你引入的依赖作为条件引入预先设置好的各种配置。


1.3.2.2、AutoConfigurationImportSelector


   @EnableAutoConfiguration注解导入了一个AutoConfigurationImportSelector自动配置类选择器,该类可以实现配置类批量载入spring容器。其核心方法是getCandidateConfigurations目的是在于候选的配置。


protected List<String> getCandidateConfigurations(
    AnnotationMetadata metadata, AnnotationAttributes attributes) {
    List<String> configurations = SpringFactoriesLoader.loadFactoryNames(
        getSpringFactoriesLoaderFactoryClass(),getBeanClassLoader());
    Assert.notEmpty(configurations, 
                    "No auto configuration classes found in META-INF/spring.factories. If you "
                    + "are using a custom packaging, make sure that file is correct.");
    return configurations;
}
复制代码


1.3.2.3、SpringFactoriesLoader


   getCandidateConfigurations方法作用是委托SpringFactoriesLoader去读取jar包中的META-INF/spring.factories文件, 并加载里面配置的自动配置对象。这是自动装配的核心。     写启动器需要遵循SpringBoot的规范,都需要编写META-INF/spring.factories,里面指定启动器的自动配置类,告诉SpringBoot需要提前放置哪些配置,等满足条件就直接加载。


2.JPG

3.JPG


1.2.3.4、RedisAutoConfiguration分析


   我们以RedisAutoConfiguration配置类分析。


4.JPG  

我们点进去看源码,可以发现有四个注解:


  1. @Configuration:表示这个类是一个配置类。
  2. @ConditionalOnWebApplication(type = Type.SERVLET):表示在满足项目的类是是Type.SERVLET类型。
  3. @ConditionalOnMissingBean({RepositoryRestMvcConfiguration.class}):表示如果环境中没有RepositoryRestMvcConfiguration这个Bean对象才生效。这个就是自定义配置的入口。
  4. @ConditionalOnClass({RepositoryRestMvcConfiguration.class}):表示满足容器中存在RepositoryRestMvcConfiguration这个Bean对象的时候才会生效。


@Configuration(
  proxyBeanMethods = false
)
@ConditionalOnWebApplication(
  type = Type.SERVLET
)
@ConditionalOnMissingBean({RepositoryRestMvcConfiguration.class})
@ConditionalOnClass({RepositoryRestMvcConfiguration.class})
@AutoConfigureAfter({HttpMessageConvertersAutoConfiguration.class, JacksonAutoConfiguration.class})
@EnableConfigurationProperties({RepositoryRestProperties.class})
@Import({RepositoryRestMvcConfiguration.class})
public class RepositoryRestMvcAutoConfiguration {
  public RepositoryRestMvcAutoConfiguration() {
  }
  @Bean
  public SpringBootRepositoryRestConfigurer springBootRepositoryRestConfigurer() {
    return new SpringBootRepositoryRestConfigurer();
  }
}
复制代码


1.4、注意


1.4.1、为什么打成war包不是jar包


   SpringBoot的默认打包方式是jar包。    在以前的开发中,Tomcat猫和web项目是独立的,必须满足一定的规则,Tomcat猫才可以部署war包。但是SpringBoot的项目是内嵌Tomcat,部署和运行一气呵成,所以就打成jar包方便。


1.4.2、pom.xml文件中的spring-boot-starter的作用


   我们在创建SpringBoot项目的时候,可以发现引入了很多的start,他收集了市面上常用的jar包以及各种依赖,并且对这些依赖进行了版本管理,避免了很多的版本冲突问题,大大简化了我们的开发,在后续的项目中我们如果需要引入某个依赖,只需引入他的start即可,依赖的版本无需引入。列举额一些SpringBoot的常用的启动器:


spring-boot-starter: 核心启动器 , 提供了自动配置,日志和YAML配置支持
spring-boot-starter-aop: 支持使用 `Spring AOP` 和 `AspectJ` 进行切面编程。
spring-boot-starter-freemarker: 支持使用 `FreeMarker` 视图构建Web 应用
spring-boot-starter-test: 支持使用 `JUnit`, 测试 `Spring Boot` 应用
spring-boot-starter-web: 支持使用 `Spring MVC` 构建 Web 应用,包括 `RESTful` 应用,使用 `Tomcat` 作为默认的嵌入式容器。
spring-boot-starter-actuator: 支持使用 Spring Boot Actuator 提供生产级别的应用程序监控和管理功能。
spring-boot-starter-logging:  提供了对日志的支持 , 默认使用Logback 
复制代码


1.4.3、mave中强大的功能——继承


   继承是 Maven 中很强大的一种功能,继承可以使得子POM可以获得 parent 中的部分配置(groupId,version,dependencies,build,dependencyManagement等),可以对子pom进行统一的配置和依赖管理。


1.4.4、DepencyManagement & dependencies区别


   在SpringBoot的pom文件中,dependencies是放在DepencyManagement中的,那么这两者的区别何在:


  1. dependencies:即使在子项目中不写该依赖项,那么子项目仍然会从父项目中继承该依赖项(全部继承)。


  1. dependencyManagement:里只是声明依赖,并不实现引入,因此子项目需要显示的声明需要用的依赖。如果不在子项目中声明依赖,是不会从父项目中继承下来的;只有在子项目中写了该依赖项,并且没有指定具体版本,才会从父项目中继承该项,并且version和scope都读取自父pom;另外如果子项目中指定了版本号,那么会使用子项目中指定的jar版本(部分继承)。


1.4.5、SpringBoot在没有Tomcat的情况下如何启动


   springboot使用嵌入式tomcat,编程实现,默认端口是8080,可以在application.properties中使用server.port进行设置。


5.JPG


1.4.6、SpringBoot的启动类的main方法中

SpringApplication.run(..)详解

   

启动类中的主方法有四个作用:


  1. 启动SpringBoot程序。
  2. 加载自定义的配置类,完成自动装配。
  3. 将当前项目部署到内嵌的Tomcat中。
  4. 启动Tomcat运行项目。


二、SpringBoot配置文件语法


2.1、SpringBoot配置文件概述


   当我们使用spring 初始化器构建完一个Spring Boot项目后,只需引入一个web启动器的依赖,它就变成一个web项目了,而我们什么都没有配置就能通过localhost:8080进行访问了,这是因为Spring Boot在底层已经把配置信息都给我们自动配置好了。     那我们怎么去修改默认配置信息?在使用Spring 初始化器创建一个Springboot项目的时候会在resources目录下自动生成一个文件 application.properties,这是一个空文件,它的作用是提供我们修改默认配置信息的入口。     Spring Boot还提供给我们另外一种风格的配置文件 application.yml,虽然是两个不同的文件但是本质是一样的,区别只是其中的语法略微不同。


2.2、Properties语法


application.properties 配置文件比较简单,他不需要空格进行区分,父属性和子属性之间是以.进行区分的。


# key = value
server.port=8080
复制代码


2.3、YML


   他是一种全新的语法格式,虽然书写略微繁琐但是他确实SpringBoot官方推荐的配置文件的格式,因为他可以存储比roperties配置文件更复杂的类型。他的语法特点:


  1. 大小写敏感。
  2. k:(空格)v:表示一对键值对(空格必须有),以空格的缩进来控制层级关系。
  3. 只要是左对齐的一列数据,则表示都是同一个层级的。
  4. “#” 表示注释,从这个字符一直到行尾,都会被解析器忽略。


server:
port: 8080
复制代码


三、SpringBoot整合


   SpringBoot最重要是整合各种框架,包括我们熟悉的Mybatis、Shiro等。


3.1、连接数据库


3.1.1、引入依赖


<!-- MySQL驱动 -->
<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
</dependency>
<!--springboot整合jdbc-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>
复制代码


3.1.2、Hikari数据源


3.1.2.1、概述


   在springboot2.0之后 , 采用的默认连接池就是Hikari, 号称"史上最快的连接池", 所以我们没有添加依赖也能直接用, springboot的自动配置中含有DataSourceAutoConfiguration配置类, 会先检查容器中是否已经有连接池对象, 没有则会使用默认的连接池, 并根据特定的属性来自动配置连接池对象, 用到的属性值来源于DataSourceProperties对象。


3.1.2.2、修改application.properties


spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.url=jdbc:mysql:///ssm_carbusiness?useUnicode=true&characterEncoding=utf8&serverTimezone=GMT%2B8
spring.datasource.username=root
spring.datasource.password=123456
复制代码


3.1.3、Druid数据源


3.1.3.1、概述


   虽然SpringBoot官方推荐的是Hikari数据源,但是如果我们想使用Druid数据源的话,也很简单。只需要添加依赖即可, 此时加的是Druid的springboot自动配置包, 里面包含了DruidDataSourceAutoConfigure自动配置类,会自动创建druid的连接池对象, 所以springboot发现已经有连接池对象了,则不会再使用Hikari


3.1.3.2、引入依赖


<!-- druid数据源依赖,记住一定要引带start的 -->
<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>druid-spring-boot-starter</artifactId>
    <version>1.2.5</version>
</dependency>
复制代码


3.1.3.3、修改application.properties


spring.datasource.druid.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.druid.url=jdbc:mysql:///ssm_carbusiness?useUnicode=true&characterEncoding=utf8&serverTimezone=GMT%2B8
spring.datasource.druid.username=root
spring.datasource.druid.password=123456
复制代码


3.1.3.4、注意


   如果我们在写代码的时候不小心引错了包,引入了普通的依赖(不带start的普通依赖,只有Druid自身的依赖),并不是自动的配置包。那么SpringBoot就不会去进行解析还是需要我们去手动配置才可以生效。我们只需在application.properties中添加一行配置即可。


spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
复制代码


3.2、集成MyBatis


3.2.1、引入依赖


<!--mybatis集成到SpringBoot中的依赖-->
<dependency>
    <groupId>org.mybatis.spring.boot</groupId>
    <artifactId>mybatis-spring-boot-starter</artifactId>
    <version>2.1.4</version>
</dependency>
复制代码


3.2.2、配置接口扫描


   在传统的SSM项目中,我们可以在配置文件中告诉Spring我的Mapper接口的位置,从而可以创建Mapper接口实现类的代理对象,在SpringBoot中没有了这个配置文件,那么我们只需在SpringBoot的启动类中添加一行配置即可。


@SpringBootApplication
// 添加这一行配置,告诉SpringBoot我的Mapper接口的位置在哪里
@MapperScan("cn.linstudy.mapper")
public class App {
  public static void main(String[] args) {
    SpringApplication.run(App.class, args);
  }
}
复制代码


3.2.3、配置属性


   在以前我们需要在application.xml中配置一些属性,从而更好地使用MyBatis,比如说mapper.xml文件的位置、是否开启懒加载、以及别名等等信息,现在这些信息都要在application.properties中进行配置,如果业务中无需使用,也可以不需要配置。


# 是否1开启懒加载
mybatis.configuration.lazy-loading-enabled=true
# 开启懒加载的方法
mybatis.configuration.lazy-load-trigger-methods=clone
# mapper.xml文件的配置
mybatis.mapper-locations=classpath:cn/wolfcode/*/mapper/*Mapper.xml
# 配置别名
mybatis.type-aliases-package=cn.wolfcode.sb.domain
#打印SQL日志
logging.level.cn.linstudy.mapper=trace
复制代码


3.3、事务管理


3.3.1、引入依赖


<!-- 支持使用 Spring AOP 和 AspectJ 进行切面编程。 -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
</dependency>
复制代码


3.3.2、XML方式配置事务


   采取配置类和XML混用的策略, 在配置类上使用@ImportResource("classpath:spring-tx.xml"),我们在这个xml文件中书写事务、切面、切入点表达式。不推荐使用


3.3.3、注解方式配置事务


   SpringBoot的自动配置中提供了TransactionAutoConfiguration事务注解自动配置类,我们在引入依赖后,直接在业务层实现类上或者其方法上直接贴@Transactional注解即可,推荐使用。切记在使用之前一定要看看数据库的引擎是否支持事务。


   SpringBoot默认使用的是CGLIB的动态代理,如果想要使用JDK的动态代理,那么仅需在application.properties中写一行配置即可。


#优先使用JDK代理
spring.aop.proxy-target-class=false 
复制代码



3.4、SpringBoot中静态资源的处理


3.4.1、静态资源的位置


SpringBoot的标准目录和1传统的SSM的maven目录是不一样的,他在resources下还有两个目录专门来存放静态资源:


  1. static:用来存放CSS、JS等样式。
  2. templates:用来存放页面模板。


6.JPG  

SpringBoot对于静态文件的处理也是有规则的:


  1. 默认情况下,Springboot会从classpath下的 /static /public/resources /META-INF/resources中四个位置下加载静态资源。


7.JPG


  1. 可以在application.properties中配置spring.resources.staticLocations属性来修改静态资源加载地址。


  1. 因为SpringBoot默认对static下静态资源的映射路径为/,所以我们引入的时候无需写static


3.4.2、路径映射配置

   在SpringBoot自动装配中,WebMvcAutoConfiguration的自动配置类导入了DispatcherServletAutoConfiguration的配置对象,会自动创建DispatcherServlet前端控制器,默认的< url-pattern >/


3.5、统一异常处理


3.5.1、SpringBoot默认方式


   SpringBoot默认情况下,会把所有错误都交给BasicErrorController类完成处理,错误的视图导向到 classpath:/static/error/ classpath:/templates/error/** 路径上,http状态码就是默认视图的名称,如果出现了404错误,那么对应的模板为404.html。     如果我们想自己写一个错误页面,那么我们只需在默认的路径下创建一个同名的模板文件即可。


3.5.2、控制器增强的方式


   自己定义一个控制器增强器,专门用于统一异常处理,该方式一般用于5xx类错误


@ControllerAdvice //控制器增强器
public class ExceptionControllerAdvice {
    @ExceptionHandler(RuntimeException.class) //处理什么类型的异常
    public String handlException(RuntimeException e, Model model) {
        return "errorView"; //错误页面视图名称
    }
}
复制代码


3.6、过滤器


3.6.1、过滤器概述


   过滤器是基于Servlet 技术实现的, 简单的来说,过滤器就是起到过滤的作用,在web项目开发中帮我们过滤一些指定的 url做一些特殊的处理,他主要的功能如下:


  1. 过滤掉一些不需要的东西,例如一些错误的请求。
  2. 可以修改请求和相应的内容。
  3. 可以拿来过滤未登录用户。


3.6.2、过滤器实现方式


   主要有两种实现方式:


  1. 第一种是使用@WebFilter
  2. 第二种是使用 FilterRegistrationBean


3.6.3、@WebFilter


3.6.3.1、概述


   @WebFilter 用于将一个类声明为过滤器,该注解将会在部署时被容器处理,容器将根据具体的属性配置将相应的类部署为过滤器。


属性名 类型 描述
filterName String 指定该Filter的名称
urlPatterns String String
value String 与 urlPatterns 一致


3.6.3.2、代码实现


创建一个MyFilter.java实现Filter接口

import org.springframework.stereotype.Component;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
@WebFilter(urlPatterns = "/api/*",filterName = "myFilter")
@Order(1)//指定过滤器的执行顺序,值越大越靠后执行
public class MyFilter implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
  System.out.println("初始化过滤器");
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse,
  FilterChain filterChain) throws IOException, ServletException {
  HttpServletRequest request= (HttpServletRequest) servletRequest;
  String uri=request.getRequestURI();
  String method=request.getMethod();
  System.out.println(uri+" "+method+"哈哈我进入了 MyFilter 过滤器了");
  filterChain.doFilter(servletRequest,servletResponse);
  }
}
复制代码

启动类加上 @ServletComponentScan 注解

创建一个 FilterController 接口

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/api")
public class HelloController {
  @GetMapping("/user/filter")
  public String hello(){
    return "哈哈我通过了过滤器";
  }
}
复制代码


3.6.4、FilterRegistrationBean 实现


创建 FilterConfig

import com.yingxue.lesson.filter.MyFilter;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class FilterConfig {
@Bean
public MyFilter myFilter(){
return new MyFilter();
}
@Bean
public FilterRegistrationBean getFilterRegistrationBean(MyFilter myFilter){
        FilterRegistrationBean filterRegistrationBean=new FilterRegistrationBean();
        /**
        * 设置过滤器
        */
        filterRegistrationBean.setFilter(MyFilter());
        /**
        * 拦截路径
        */
        filterRegistrationBean.addUrlPatterns("/api/*");
        /**
        * 设置名称
        */
        filterRegistrationBean.setName("myFilter");
        /**
        * 设置访问优先级 值越小越高
        */
        filterRegistrationBean.setOrder(1);
  return filterRegistrationBean;
  }
}
复制代码


修改 MyFilter.java

//@WebFilter(urlPatterns ={"/api/*"},filterName = "myFilter")
复制代码


修改启动类

//@ServletComponentScan
复制代码


3.7、拦截器


3.7.1、概述


   简单的来说,就是一道阀门,在某个方法被访问之前,进行拦截,然后在之前或之后加入某些操作,拦截器是AOP 的一种实现策略,他的主要功能是对正在运行的流程进行干预。


3.7.2、拦截器方法概述


   拦截器也主要有三个方法:


  1. preHandle是在请求之前就进行调用,如果该请求需要被拦截,则返回false,否则true。
  2. postHandle是在请求之后进行调用,无返回值。
  3. afterCompletion是在请求结束的时候进行调用,无返回值。


3.7.3、代码实现


创建拦截器类并实现 HandlerInterceptor 接口

public class MyInterceptor implements HandlerInterceptor {
  @Value("${open.url}")
  private String openUrl;
  @Override
  public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    System.out.println("MyInterceptor....在请求处理之前进行调用(Controller方法调用之前)");
    String requestUrl=request.getRequestURI();
    System.out.println("过滤器MyFilter拦截了请求为"+requestUrl);
    return true;
  }
    @Override
  public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
    System.out.println("MyInterceptor...请求处理之后进行调用,但是在视图被渲染之前(Controller方法调用之后)");
  }
  @Override
  public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
    System.out.println("MyInterceptor....在整个请求结束之后被调用,也就是在DispatcherServlet 渲染了对应的视图之后执行(主要是用于进行资源清理工作)");
  }
}
复制代码


修改 application.properties

 

我们需要在application.properties中添加一行代码加入开发接口通配地址,表示放行的代码


#凡是请求地址层级带有 open 都放行
open.url=/**/open/**
复制代码

创建一个 Java 实现 WebMvcConfigurer,并重写 addInterceptors 方法

@Configuration
public class WebAppConfig implements WebMvcConfigurer {
  @Value("${open.url}")
  private String openUrl;
  @Bean
  public MyInterceptor getMyInterceptor(){
    return new MyInterceptor();
  }
  @Override
  public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(getMyInterceptor()).addPathPatterns("/api/**").excludePathPatterns(openUrl);
  }
}
复制代码


3.7.4、测试


@RestController
@RequestMapping("/api")
public class InterceptorController {
  @GetMapping("/home/open/info")
  public String home(){
    return "欢迎来到首页";
  }
  @GetMapping("/user/interceptor")
  public String interceptor(){
    return "我被拦截了并通过了拦截器";
  }
}
复制代码


3.8、日志


   日志对于我们的系统测试和调试时有着很重要的地位,我们在系统开发的时候,以前时经常使用System.out.print()这句话来输出一些系统的信息,但是在实际的工作中却不会用到这种方法来输出日志,原因大概有以下几点:


  1. 比起System.out.println,日志框架更为灵活,可以把日志的输出和代码分离。
  2. 日志框架可以方便的定义日志的输出环境,控制台,文件,数据库。
  3. 日志框架可以方便的定义日志的输出格式和输出级别。


3.8.1、日志介绍


3.8.1.1、SpringBoot中的日志介绍


   我们在SpringBoot启动的时候就可以看到时默认开启了日志的。从左往右分别为:时间日志级别线程ID线程名称日志类日志说明

12.JPG

   日志级别,级别越高,输出的内容越少, 如果设置的级别为info, 则debug以及trace级别的都无法显示trace < debug < info < warn < error     Springboot默认选择Logback作为日志框架,也能选择其他日志框架,但是没有必要

8.JPG


3.8.1.2、输出日志的两种方式


在类中定义一个静态Logger对象


// 这里传入当前类的作用是方便输出日志时可以清晰地看到该日志信息是属于哪个类的
private static final Logger log = LoggerFactory.getLogger(当前类.class);
复制代码


使用lombok提供的@Slf4j注解


@Slf4j
@Service
public class PermissionServiceImpl implements IPermissionService {}
    //输出日志中有变量可以使用{}作为占位符
    log.info("删除id为{}的数据", id);
    log.debug("权限插入成功:{}",expression);
    log.info("权限插入成功:{}",expression);
    log.warn("权限插入成功:{}",expression);
}
复制代码


3.8.1.3、Logback配置文件的使用


   Logback框架默认会自动加载classpath:logback.xml,作为框架的配置文件。在SpringBoot中使用时,还会额外的支持自动加载classpath:logback-spring.xml。所以推荐使用logback-spring.xml,功能更强大些。


<?xml version="1.0" encoding="UTF-8"?>
<!--
    scan:开启日志框架的热部署,默认值true表示开启
    scanPeriod:热部署的频率,默认值60 second
    debug:设置输出框架内部的日志,默认值false
-->
<configuration scan="true" scanPeriod="60 second" debug="false">
    <property name="appName" value="springboot demo" />
    <contextName>${appName}</contextName>
    <!-- appender:日志输出对象,配置不同的类拥有不同的功能
        ch.qos.logback.core.ConsoleAppender:日志输出到控制台        
    -->
    <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
        <encoder>
            <pattern>%d{yyyy-MM-dd-HH:mm:ss} %level [%thread]-%logger{35} >> %msg %n</pattern>
     </encoder>
   </appender>
    <!-- ch.qos.logback.core.FileAppender:日志输出到文件中
    <appender name="fileAppender" class="ch.qos.logback.core.FileAppender">
        <encoder>
            <pattern>%-4relative [%thread] %level %logger{35} - %msg %n</pattern>
       </encoder>
        <append>true</append>
        <file>mylog.log</file>
   </appender>
    -->
    <!-- root是项目通用的logger,一般情况下都是使用root配置的日志输出
        level:按照级别输出日志,日志级别,级别越高,输出的内容越少
            trace < debug < info < warn < error
    -->
   <root level="info">
        <appender-ref ref="STDOUT" />
    </root>
    <!-- 自定义的logger,用于专门输出特定包中打印的日志
    <logger name="cn.wolfcode.crm.mapper" level="trace">
    </logger>
  -->
</configuration>
复制代码


3.9、集成前端


3.9.1、集成JSP


   提起 Java 不得不说的一个开发场景就是 Web 开发,说到 Web 开发绕不开的一个技术就是 JSP,SpringBoot官方虽然已经不推荐使用JSP了,但是集成JSP还是很重要的。


3.9.1.1、引入依赖


<!--JSP标准标签库-->
<dependency>
  <groupId>javax.servlet</groupId>
  <artifactId>jstl</artifactId>
</dependency>
<!--内置tocat对Jsp支持的依赖,用于编译Jsp-->
<dependencys>
  <groupId>org.apache.tomcat.embed</groupId>
  <artifactId>tomcat-embed-jasper</artifactId>
</dependency>
复制代码


3.9.1.2、编辑 project Structure


9.JPG


3.9.1.3、Spring Mvc 视图解析器配置


   我们需要修改application .properties,加入Spring Mvc 视图解析器配置


spring.mvc.view.prefix=/WEB-INF/jsp/
spring.mvc.view.suffix=.jsp
复制代码


3.9.1.4、总结


   所以我们在以后遇到,老旧的项目升级成Spring Boot 项目时候,首先得配置好 webapp 这个跟路径、配置好 web、再配置 ORM 所需的一些配置,最后记得配置视图解析器。所需的配置配置好后就可以直接把代码拷入新的项目了。


3.9.2、集成FreeMarker


   在传统的SpringMVC中集成FreeMarker需要把FreeMarkerConfigurer和FreeMarkerViewResolve两个对象配置到Spring容器中,同时两个对象都要分别配置一些属性,还是比较麻烦的,在SpringBoot中,依靠自动配置功能,我们可以非常轻松的实现集成FreeMarker,只需要引入一个依赖即可。


<!-- SpringBoot集成FreeMarker的依赖 -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-freemarker</artifactId>
</dependency>
复制代码


3.9.2.1、底层原理


   SpringBoot的自动配置中含有FreeMarkerAutoConfiguration配置对象,该配置对象又导入了FreeMarkerReactiveWebConfiguration配置对象,在里面创建了FreeMarkerConfigurerFreeMarkerViewResolve两个对象交给Spring管理,并且设置了默认的属性值,这些属性值来源于FreeMarkerProperties对象


3.9.2.2、常见属性配置


# 是否开启freemarker支持
spring.freemarker.enabled=true
# 模板编码
spring.freemarker.charset=UTF-8
# 模板contenttype
spring.freemarker.content-type=text/html
# 是否开启session属性暴露,默认false
spring.freemarker.expose-session-attributes = false
# 加载模板时候的前缀
spring.freemarker.prefix: templates
# 模板文件后缀,SpringBoot2.X默认是ftlh,有时候我们的模板后缀是ftl
spring.freemarker.suffix: ftl
# 模板加载地址
spring.freemarker.template-loader-path=classpath:/templates/
#一般我们会做3个配置,其余默认  
# 暴露session对象的属性
spring.freemarker.expose-session-attributes=true
# 配置为传统模式,空值自动处理
spring.freemarker.settings.classic_compatible=true
# 重新指定模板文件后缀 springboot 2.2.x 后 默认后缀为 .ftlh
spring.freemarker.suffix=.ftl  
复制代码


3.9.3、整合 Thymeleaf


   Thymeleaf是一款用于渲染XML/XHTML/HTML5内容的模板引擎。类似JSP,FreeMaker等, 它也可以轻易的与 Web 框架进行集成作。     为 Web 应用的模板引擎。与其它模板引擎相比, Thymeleaf 最大的特点是能够直接在浏览器中打开并正确显示模板页面,而不需要启动整个Web应用。 thymeLeaf支持Spring Expression Language语言作为方言,也就是SpEL,SpEL是可以用于Spring中的一种EL表达式。


   它与我们使用过的JSP不同,thymeleaf是使用html的标签来完成逻辑和数据的传入进行渲染。可以说用 thymeleaf 完全替代 jsp 是可行的。


3.9.3.1、创建项目


   我们在创建项目的时候记得勾选这两个依赖选项。

10.JPG

11.JPG


3.9.3.2、Spring Mvc 视图解析器配置


#thymeleaf
# 前缀 默认读取classpath:/templates/
#无需配置
#spring.thymeleaf.prefix=classpath:/templates/
# 后缀
spring.thymeleaf.suffix=.html
spring.thymeleaf.charset=UTF-8
spring.thymeleaf.servlet.content-type=text/html
复制代码


3.9.3.3、测试


   在templates下创建一个hello.html 必须加入xmlns:th="www.thymeleaf.org" Thymeleaf声明空间。

相关文章
|
2月前
|
人工智能 自然语言处理 前端开发
SpringBoot + 通义千问 + 自定义React组件:支持EventStream数据解析的技术实践
【10月更文挑战第7天】在现代Web开发中,集成多种技术栈以实现复杂的功能需求已成为常态。本文将详细介绍如何使用SpringBoot作为后端框架,结合阿里巴巴的通义千问(一个强大的自然语言处理服务),并通过自定义React组件来支持服务器发送事件(SSE, Server-Sent Events)的EventStream数据解析。这一组合不仅能够实现高效的实时通信,还能利用AI技术提升用户体验。
245 2
|
4月前
|
开发框架 负载均衡 Java
当热门技术负载均衡遇上 Spring Boot,开发者的梦想与挑战在此碰撞,你准备好了吗?
【8月更文挑战第29天】在互联网应用开发中,负载均衡至关重要,可避免单服务器过载导致性能下降或崩溃。Spring Boot 作为流行框架,提供了强大的负载均衡支持,通过合理分配请求至多台服务器,提升系统可用性与可靠性,优化资源利用。本文通过示例展示了如何在 Spring Boot 中配置负载均衡,包括添加依赖、创建负载均衡的 `RestTemplate` 实例及服务接口调用等步骤,帮助开发者构建高效、稳定的应用。随着业务扩展,掌握负载均衡技术将愈发关键。
125 6
|
23天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
2月前
|
存储 Java API
简单两步,Spring Boot 写死的定时任务也能动态设置:技术干货分享
【10月更文挑战第4天】在Spring Boot开发中,定时任务通常通过@Scheduled注解来实现,这种方式简单直接,但存在一个显著的限制:任务的执行时间或频率在编译时就已经确定,无法在运行时动态调整。然而,在实际工作中,我们往往需要根据业务需求或外部条件的变化来动态调整定时任务的执行计划。本文将分享一个简单两步的解决方案,让你的Spring Boot应用中的定时任务也能动态设置,从而满足更灵活的业务需求。
183 4
|
2月前
|
存储 JSON 算法
JWT令牌基础教程 全方位带你剖析JWT令牌,在Springboot中使用JWT技术体系,完成拦截器的实现 Interceptor (后附源码)
文章介绍了JWT令牌的基础教程,包括其应用场景、组成部分、生成和校验方法,并在Springboot中使用JWT技术体系完成拦截器的实现。
147 0
JWT令牌基础教程 全方位带你剖析JWT令牌,在Springboot中使用JWT技术体系,完成拦截器的实现 Interceptor (后附源码)
|
3月前
|
存储 缓存 Java
在Spring Boot中使用缓存的技术解析
通过利用Spring Boot中的缓存支持,开发者可以轻松地实现高效和可扩展的缓存策略,进而提升应用的性能和用户体验。Spring Boot的声明式缓存抽象和对多种缓存技术的支持,使得集成和使用缓存变得前所未有的简单。无论是在开发新应用还是优化现有应用,合理地使用缓存都是提高性能的有效手段。
50 1
|
2月前
|
机器学习/深度学习 移动开发 自然语言处理
基于人工智能技术的智能导诊系统源码,SpringBoot作为后端服务的框架,提供快速开发,自动配置和生产级特性
当身体不适却不知该挂哪个科室时,智能导诊系统应运而生。患者只需选择不适部位和症状,系统即可迅速推荐正确科室,避免排错队浪费时间。该系统基于SpringBoot、Redis、MyBatis Plus等技术架构,支持多渠道接入,具备自然语言理解和多输入方式,确保高效精准的导诊体验。无论是线上医疗平台还是大型医院,智能导诊系统均能有效优化就诊流程。
|
4月前
|
NoSQL JavaScript 前端开发
SpringBoot+Vue实现校园二手系统。前后端分离技术【完整功能介绍+实现详情+源码】
文章介绍了如何使用SpringBoot和Vue实现一个校园二手系统,采用前后端分离技术。系统具备完整的功能,包括客户端和管理员端的界面设计、个人信息管理、商品浏览和交易、订单处理、公告发布等。技术栈包括Vue框架、ElementUI、SpringBoot、Mybatis-plus和Redis。文章还提供了部分源代码,展示了前后端的请求接口和Redis验证码功能实现,以及系统重构和模块化设计的一些思考。
SpringBoot+Vue实现校园二手系统。前后端分离技术【完整功能介绍+实现详情+源码】
|
4月前
|
缓存 NoSQL Java
SpringBoot的三种缓存技术(Spring Cache、Layering Cache 框架、Alibaba JetCache 框架)
Spring Cache 是 Spring 提供的简易缓存方案,支持本地与 Redis 缓存。通过添加 `spring-boot-starter-data-redis` 和 `spring-boot-starter-cache` 依赖,并使用 `@EnableCaching` 开启缓存功能。JetCache 由阿里开源,功能更丰富,支持多级缓存和异步 API,通过引入 `jetcache-starter-redis` 依赖并配置 YAML 文件启用。Layering Cache 则提供分层缓存机制,需引入 `layering-cache-starter` 依赖并使用特定注解实现缓存逻辑。
1265 1
SpringBoot的三种缓存技术(Spring Cache、Layering Cache 框架、Alibaba JetCache 框架)
|
4月前
|
小程序 前端开发 JavaScript
【项目实战】SpringBoot+uniapp+uview2打造一个企业黑红名单吐槽小程序
【避坑宝】是一款企业黑红名单吐槽小程序,旨在帮助打工人群体辨别企业优劣。该平台采用SpringBoot+MybatisPlus+uniapp+uview2等技术栈构建,具备丰富的注释与简洁的代码结构,非常适合实战练习与学习。通过小程序搜索“避坑宝”即可体验。
124 0
【项目实战】SpringBoot+uniapp+uview2打造一个企业黑红名单吐槽小程序