当前位置:   article > 正文

Spingboot3详解(全网最详细,新建springboot项目并详解各种组件的用法)_springboot3

springboot3

一.Spring Initializr创建向导

1.新建一个空项目

 2.在新创建的空项目里,新建Module

3. 选择Spring Initializr

4.选择Spring Boot的版本3+以上

5.创建好的一个项目结构

controller包是自己创建的

6.项目结构分析

spingboot主程序

  1. package com.example.boot;
  2. import org.springframework.boot.SpringApplication;
  3. import org.springframework.boot.autoconfigure.SpringBootApplication;
  4. @SpringBootApplication //加上这个注解,表示这是一个SpringBoot应用
  5. public class Boot302DemoApplication {
  6. public static void main(String[] args) {
  7. SpringApplication.run(Boot302DemoApplication.class, args);
  8. }
  9. }

7.测试

com.example.boot下创建controller包-再在该包下创建一个HelloController类

@ResponseBody  //获取的是文本,而不是跳转页面的

@Controller

这两个注解相当于一个@RestController

运行主类

 运行结果:

在浏览器中输入localhost:8080/haha

以上一个简单的Springboot项目测试完成 

二.组件注册

SpringBoot摒弃XML配置方式,改为全注解驱动

@Configuration@SpringBootConfiguration

@Bean@Scope

@Controller、 @Service、@Repository、@Component

@Import

@ComponentScan

1.@Configuration与 @Bean的使用

(1)组件在容器中的名字默认是方法名

举例:

新建bean包,再包下创建User和Cat两个类

新建config包,再该包下创建AppConfig类

User和Cat类下的内容

User类

  1. package com.example.boot.bean;
  2. public class User {
  3. private Long id;
  4. private String name;
  5. public Long getId() {
  6. return id;
  7. }
  8. public void setId(Long id) {
  9. this.id = id;
  10. }
  11. public String getName() {
  12. return name;
  13. }
  14. public void setName(String name) {
  15. this.name = name;
  16. }
  17. }

Cat类

  1. package com.example.boot.bean;
  2. public class Cat {
  3. private Long id;
  4. private String name;
  5. public Long getId() {
  6. return id;
  7. }
  8. public void setId(Long id) {
  9. this.id = id;
  10. }
  11. public String getName() {
  12. return name;
  13. }
  14. public void setName(String name) {
  15. this.name = name;
  16. }
  17. }

AppConfig类

//组件在容器中的名字默认是方法名

  1. package com.example.boot.config;
  2. import com.example.boot.bean.User;
  3. import org.springframework.context.annotation.Bean;
  4. import org.springframework.context.annotation.Configuration;
  5. /**
  6. * @description
  7. * @author: Admin
  8. * @create: 2023/12/7 9:35
  9. */
  10. @Configuration //这是一个配置类,代替之前的配置文件
  11. public class AppConfig {
  12. //组件在容器中的名字是方法名
  13. @Bean //该注解替代之前的bean标签,就是在ioc.xml里的配置的标签
  14. public User user01(){
  15. var user=new User();
  16. user.setId(1L);
  17. user.setName("小苏");
  18. return user;
  19. }
  20. }

ioc.xml里的配置信息

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
  5. <bean id="user" class="com.example.boot.bean.User">
  6. <property name="id" value="1"></property>
  7. <property name="name" value="zhangsan"></property>
  8. </bean>
  9. <bean id="cat" class="com.example.boot.bean.Cat">
  10. <property name="id" value="1"></property>
  11. <property name="name" value="Tom"></property>
  12. </bean>
  13. </beans>

 运行主类:

再运行结果中查询user01

(2)组件在容器中的名字默认是方法名,但是可以直接修改注解的值

举例:

运行主类,再主类的结果中查询hahaha

2.@Scope应用

组件默认都是单实例),想要变成多实例(每次获取都会重新创建对象),可添加@Scope(“prototype”)

3.@SpringBootConfiguration

与@Configuration注解差不多

 4.@Import

@Import也是放在配置类中的

三.条件注解

如果注解指定的条件成立,则触发指定行为

@ConditionalOnXxx

举例常见的

@ConditionalOnClass:如果类路径中存在这个类,则触发指定行为

@ConditionalOnMissingClass:如果类路径中不存在这个类,则触发指定行为

@ConditionalOnBean:如果容器中存在这个Bean(组件),则触发指定行为

@ConditionalOnMissingBean:如果容器中不存在这个Bean(组件),则触发指定行为

四.属性绑定 

@ConfigurationProperties: 声明组件的属性和配置文件哪些前缀开始项进行绑定

@EnableConfigurationProperties:快速注册注解:

  • 场景:SpringBoot默认只扫描自己主程序所在的包。如果导入第三方包,即使组件上标注了 @Component、@ConfigurationProperties 注解,也没用。因为组件都扫描不进来,此时使用这个注解就可以快速进行属性绑定并把组件注册进容器

将容器中任意组件(Bean)的属性值配置文件的配置项的值进行绑定

  • 1、给容器中注册组件(@Component、@Bean)
  • 2、使用@ConfigurationProperties 声明组件和配置文件的哪些配置项进行绑定

举例:

在配置文件(application.properties)中加入如下:

新建一个Pig类

Pig类

  • 1、给容器中注册组件(@Component或者@Bean)
  • 2、使用@ConfigurationProperties 声明组件和配置文件的哪些配置项进行绑定
  1. package com.example.boot.bean;
  2. import org.springframework.boot.context.properties.ConfigurationProperties;
  3. import org.springframework.stereotype.Component;
  4. /**
  5. * @description
  6. * @author: Admin
  7. * @create: 2023/12/7 14:16
  8. */
  9. @ConfigurationProperties(prefix = "pig") //prefix后面的名字是application.properties里配置里自己写的
  10. @Component
  11. public class Pig {
  12. private Long id;
  13. private String name;
  14. private Integer age;
  15. public Long getId() {
  16. return id;
  17. }
  18. public void setId(Long id) {
  19. this.id = id;
  20. }
  21. public String getName() {
  22. return name;
  23. }
  24. public void setName(String name) {
  25. this.name = name;
  26. }
  27. public Integer getAge() {
  28. return age;
  29. }
  30. public void setAge(Integer age) {
  31. this.age = age;
  32. }
  33. @Override
  34. public String toString() {
  35. return "Pig{" +
  36. "id=" + id +
  37. ", name='" + name + '\'' +
  38. ", age=" + age +
  39. '}';
  40. }
  41. }

运行主类结果:

 (乱码解决方法:File->Settings->File encofings中的编码格式都改为utf-8)

 五.自动配置机制

1. 初步理解

  • 自动配置的 Tomcat、SpringMVC 等
    • 导入场景,容器中就会自动配置好这个场景的核心组件。
    • 以前:DispatcherServlet、ViewResolver、CharacterEncodingFilter....
    • 现在:自动配置好的这些组件
    • 验证:容器中有了什么组件,就具有什么功能
  1. public static void main(String[] args) {
  2. //局部变量类型的自动推断
  3. var ioc = SpringApplication.run(MainApplication.class, args);
  4. //1、获取容器中所有组件的名字
  5. String[] names = ioc.getBeanDefinitionNames();
  6. //2、挨个遍历:
  7. // dispatcherServlet、beanNameViewResolver、characterEncodingFilter、multipartResolver
  8. // SpringBoot把以前配置的核心组件现在都给我们自动配置好了。
  9. for (String name : names) {
  10. System.out.println(name);
  11. }
  12. }
  • 默认的包扫描规则
    • @SpringBootApplication 标注的类就是主程序类
    • SpringBoot只会扫描主程序所在的包及其下面的子包,自动的component-scan功能自定义扫描路径
    • @SpringBootApplication(scanBasePackages = "com.atguigu")
    • @ComponentScan("com.atguigu") 直接指定扫描的路径
  • 配置默认值
    • 配置文件的所有配置项是和某个类的对象值进行一一绑定的。
    • 绑定了配置文件中每一项值的类: 属性类
  • 比如:
    • ServerProperties绑定了所有Tomcat服务器有关的配置
    • MultipartProperties绑定了所有文件上传相关的配置
    • ....参照官方文档:或者参照 绑定的 属性类
  • 按需加载自动配置
    • 导入场景spring-boot-starter-web
    • 场景启动器除了会导入相关功能依赖,导入一个spring-boot-starter,是所有starterstarter,基础核心starter
    • spring-boot-starter导入了一个包 spring-boot-autoconfigure。包里面都是各种场景的AutoConfiguration自动配置类
    • 虽然全场景的自动配置都在 spring-boot-autoconfigure这个包,但是不是全都开启的。
      • 导入哪个场景就开启哪个自动配置

总结: 导入场景启动器、触发 spring-boot-autoconfigure这个包的自动配置生效、容器中就会具有相关场景的功能

2.完整流程

思考:

1、SpringBoot怎么实现导一个starter、写一些简单配置,应用就能跑起来,我们无需关心整合

2、为什么Tomcat的端口号可以配置在application.properties中,并且Tomcat能启动成功?

3、导入场景后哪些自动配置能生效

 

自动配置流程细节梳理:

1、导入starter-web:导入了web开发场景

  • 1、场景启动器导入了相关场景的所有依赖:starter-jsonstarter-tomcatspringmvc
  • 2、每个场景启动器都引入了一个spring-boot-starter,核心场景启动器。
  • 3、核心场景启动器引入了spring-boot-autoconfigure包。
  • 4、spring-boot-autoconfigure里面囊括了所有场景的所有配置。
  • 5、只要这个包下的所有类都能生效,那么相当于SpringBoot官方写好的整合功能就生效了。
  • 6、SpringBoot默认却扫描不到 spring-boot-autoconfigure下写好的所有配置类。(这些配置类给我们做了整合操作),默认只扫描主程序所在的包

2、主程序@SpringBootApplication

  • 1、@SpringBootApplication由三个注解组成@SpringBootConfiguration@EnableAutoConfiguratio@ComponentScan
  • 2、SpringBoot默认只能扫描自己主程序所在的包及其下面的子包,扫描不到 spring-boot-autoconfigure包中官方写好的配置类
  • 3、@EnableAutoConfiguration:SpringBoot 开启自动配置的核心
    • 1. 是由@Import(AutoConfigurationImportSelector.class)提供功能:批量给容器中导入组件。
    • 2. SpringBoot启动会默认加载 142个配置类。
    • 3. 这142个配置类来自于spring-boot-autoconfigureMETA-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports文件指定的
    • 项目启动的时候利用 @Import 批量导入组件机制把 autoconfigure 包下的142 xxxxAutoConfiguration类导入进来(自动配置类
    • 虽然导入了142个自动配置类
  • 4、按需生效:
    • 并不是这142个自动配置类都能生效
    • 每一个自动配置类,都有条件注解@ConditionalOnxxx,只有条件成立,才能生效

3、xxxxAutoConfiguration自动配置类

  • 1、给容器中使用@Bean 放一堆组件。
  • 2、每个自动配置类都可能有这个注解@EnableConfigurationProperties(ServerProperties.class),用来把配置文件中配的指定前缀的属性值封装到 xxxProperties属性类
  • 3、以Tomcat为例:把服务器的所有配置都是以server开头的。配置都封装到了属性类中。
  • 4、给容器中放的所有组件的一些核心参数,都来自于xxxPropertiesxxxProperties都是和配置文件绑定。
  • 只需要改配置文件的值,核心组件的底层参数都能修改

4、写业务,全程无需关心各种整合(底层这些整合写好了,而且也生效了)

核心流程总结:

1、导入starter,就会导入autoconfigure包。

2、autoconfigure 包里面 有一个文件 META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports,里面指定的所有启动要加载的自动配置类

3、@EnableAutoConfiguration 会自动的把上面文件里面写的所有自动配置类都导入进来。xxxAutoConfiguration 是有条件注解进行按需加载

4、xxxAutoConfiguration给容器中导入一堆组件,组件都是从 xxxProperties中提取属性值

5、xxxProperties又是和配置文件进行了绑定

效果:导入starter、修改配置文件,就能修改底层行为。

3. 如何学好SpringBoot

框架的框架、底层基于Spring。能调整每一个场景的底层行为。100%项目一定会用到底层自定义

摄影:

  • 傻瓜:自动配置好。
  • 单反:焦距、光圈、快门、感光度....
  • 傻瓜+单反
  1. 理解自动配置原理
  • 导入starter --> 生效xxxxAutoConfiguration --> 组件 --> xxxProperties --> 配置文件
  1. 理解其他框架底层
  • 拦截器
  1. 可以随时定制化任何组件
  • 配置文件
  • 自定义组件

普通开发:导入starter,Controller、Service、Mapper、偶尔修改配置文件

高级开发:自定义组件、自定义配置、自定义starter

核心:

  • 这个场景自动配置导入了哪些组件,我们能不能Autowired进来使用
  • 能不能通过修改配置改变组件的一些默认参数
  • 需不需要自己完全定义这个组件
  • 场景定制化

最佳实战

  • 选场景,导入到项目
    • 官方:starter
    • 第三方:去仓库搜
  • 写配置,改配置文件关键项
    • 数据库参数(连接地址、账号密码...)
  • 分析这个场景给我们导入了哪些能用的组件
    • 自动装配这些组件进行后续使用
    • 不满意boot提供的自动配好的默认组件
      • 定制化
        • 改配置
        • 自定义组件

整合redis:

  • 选场景spring-boot-starter-data-redis
    • 场景AutoConfiguration 就是这个场景的自动配置类
  • 写配置:
    • 分析到这个场景的自动配置类开启了哪些属性绑定关系
    • @EnableConfigurationProperties(RedisProperties.class)
    • 修改redis相关的配置
  • 分析组件:
    • 分析到 RedisAutoConfiguration 给容器中放了 StringRedisTemplate
    • 给业务代码中自动装配 StringRedisTemplate
  • 定制化
    • 修改配置文件
    • 自定义组件,自己给容器中放一个 StringRedisTemplate

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/凡人多烦事01/article/detail/504812
推荐阅读
相关标签
  

闽ICP备14008679号