SpringBoot | 1.2 全注解下的Spring IoC

简介: 在学习SpringBoot之前,有几个Spring的重要的基础概念需要提一下,SpringBoot对这些基础概念做进一步的封装,完成自动配置。首先就是Spring的控制反转IOC,由于SpringBoot不建议使用XML,因此本小节旨在关注全注解下的Spring IoC。其中比较重要的是第二点与第三点:第二点里的注解要么常用于业务开发,要么在源码里常见;第四点的`@Autowired`需要关注。注:在说明注解时,第一点加粗为注解中文含义,第二点为一般加在哪身上,缩进或代码块为示例,如:**@注解** - **中文含义** - 加在哪 - 其他…… - `语句示例`

1. Ioc容器概念

Spring的IoC容器其实就是一个Bean管理的容器。

所有的IoC容器都要实现顶层接口BeanFactory(Bean工厂),该工厂定义了一些操作Bean实例的基本方法,源码如下:

public interface BeanFactory {
    //前缀
    String FACTORY_BEAN_PREFIX = "&";
    
    //根据类型、名称等获取Bean
    Object getBean(String var1) throws BeansException;

    <T> T getBean(String var1, Class<T> var2) throws BeansException;

    Object getBean(String var1, Object... var2) throws BeansException;

    <T> T getBean(Class<T> var1) throws BeansException;

    <T> T getBean(Class<T> var1, Object... var2) throws BeansException;
    
    <T> ObjectProvider<T> getBeanProvider(Class<T> var1);

    <T> ObjectProvider<T> getBeanProvider(ResolvableType var1);

    //是否包含Bean
    boolean containsBean(String var1);
    
    //Bean是否单例
    boolean isSingleton(String var1) throws NoSuchBeanDefinitionException;
    
    //Bean是否原型
    boolean isPrototype(String var1) throws NoSuchBeanDefinitionException;

    //是否类型匹配
    boolean isTypeMatch(String var1, ResolvableType var2) throws NoSuchBeanDefinitionException;

    boolean isTypeMatch(String var1, Class<?> var2) throws NoSuchBeanDefinitionException;

    //获取Bean类型
    @Nullable
    Class<?> getType(String var1) throws NoSuchBeanDefinitionException;

    //获取Bean别名
    @Nullable
    Class<?> getType(String var1, boolean var2) throws NoSuchBeanDefinitionException;

    String[] getAliases(String var1);
}

从上述源码可知,顶层接口定义了一些基本方法,其中需要注意的点有:

  • isSingleton方法判断Bean是否在Spring IoC中为单例。默认情况下Bean都是以单例存在,即:调用getBean方法返回同一个对象;(详情见第五点Bean的作用域)
  • isPrototype方法判断原型,若返回true,则调用getBean方法时IoC会创建一个新的Bean返回,与Bean的作用域相关;(详情见第五点Bean的作用域)

BeanFactory基础上,有一些高级接口,可以用个实现接口完成定制化操作。

Spring IoC接口设计如下:


20210613152305811.png

需要关注ApplicationContext接口,其的实现类有:

  • FileSystemXmlApplicationContext: 表示从文件绝对路径加载配置文件;
  • ClassPathXmlApplicationContext:表示 从classpath下加载配置文件(适合于相对路径方式加载) 。
//Spring里常用获取Bean的方法
@Test
public void testBean() {
    ApplicationContext context = new ClassPathXmlApplicationContext("bean.xml");
    Orders orders = context.getBean("orders", Orders.class);
}


2. 往IoC容器添加组件

通常意义上讲的Spring IoC控制反转就是往容器中添加组件过程,针对添加的组件类型不同,有相应不同的注解实现。该点下很多注解在源码里常见。

@Repository、@Service、@Controller、@RestController、@Component、@Bean

注解 功能介绍
@Repository 数据库层组件,常用于DAO层;
@Service 业务逻辑组件,常用于业务逻辑层;
@Controller 控制器,常用于页面展示层;控制器默认方法返回的是页面跳转;
@RestController REST风格控制器, 常用于响应字符串,是@ResponseBody和@Controller的组合注解;默认方法返回的是json数据;
@Component 注册组件,常用于POJO层。可以配置Bean的名称,如@Component("XXX");不配置时Ioc容器将类名第一个字母小写,其他不变放入容器中;
@Bean 给容器注册组件。常用于第三方包,源码里常见。


@ComponentScan

  • 开启注解扫描
  • 常用于配置类;
  • 用于开启注解扫描,只会扫描当前包和其子包下的注解,可以自定义包扫描路径;
  • 可以通过excludeFilters配置指定哪些类型或类不被扫描进IoC容器;

    • 自定义路径:@ComponentScan(basePackages = {"com.dlhjw"})
    • 排除扫描:@ComponentScan(basePackages = "com.dlhjw.*", excludeFilters = {@Filter{classes = {Service.class}}})


@Import

  • 导入组件
  • 常用于配置类;
  • 用来导入配置类或者一些需要前置加载的类,导入指定类型组件;
  • 源码里常见。


@Conditional

  • 条件装配
  • 常用于配置类;
  • 对满足Conditional指定的条件,则进行组件注入;
  • 源码里常见。


20210613154010100.png

3. Full模式(单实例)与Lite模式(原型)

Full模式与Lite模式针对spring配置而言的,和xml配置无关。单实例可以理解成每次从IoC容器中取出同一个对象,原型则对应每次取出时都会new一个新对象。

Full模式(默认,单实例):

  • 标注有@Configuration或者@Configuration(proxyBeanMethods = true)的类被称为Full模式的配置类;

    • proxyBeanMethods:代理bean的方法;
  • 单实例是IoC容器启动时就会去实例化bean并添加到容器当中去,每次获取都是从容器中获取同一个对象;
  • 单例模式能有效避免Lite模式下的错误。性能没有Lite模式好;

Lite模式(多实例):

  • 类上有@Component注解;
  • 类上有@ComponentScan注解;
  • 类上有@Import注解;
  • 类上有@ImportResource注解;
  • 类上没有任何注解,但是类中存在@Bean方法;
  • 类上有@Configuration(proxyBeanMethods = false)注解;
  • 多实例是去获取对象的时候才回去实例化bean,每次获取都会去实例化bean;
  • 运行时不用生成CGLIB子类,提高运行性能,降低启动时间,可以作为普通类使用。但是不能声明@Bean之间的依赖。


4. 依赖注入DI

依赖注入讨论的是Bean之间的依赖关系,例如一个类里包含另一个类。上面第二点讨论的是添加组件,如果把它理解成创建对象的话,那么依赖注入就是获取这个对象并使用。依赖注入的前提是往容器添加组件,也就是说只有容器中有这个组件,才能取出来使用。

@Autowired

  • 自动注入
  • 常用在属性、set方法与方法的参数上;
  • 最常用的注解之一,根据属性类型(by type)找到对应的Bean,如果对应的Bean不是唯一的,则会根据其属性名称和Bean的名称匹配;
  • 其默认必须找到对应的Bean;如果不能确定其标注属性是否存在,并且允许这个被标注的属性为null,可以配置@Autowired属性required为false,如:

    • @Autowired(required = false)


@Primary

  • 优先注入
  • 常用于类或组件上;
  • 当获取的Bean不是唯一时,优先注入被@Primary标注的类或组件;


@Quelifier

  • 限定注入
  • 与@Autowired一同使用;
  • 配置需要注入的名称即可;

    @Autowired //根据类型Animal
    @Quelifier("dog") //根据名称dog
    private Animal animal;


5. Bean的生命周期

大致分为Bean的定义、Bean的初始化、Bean的生存期和Bean的销毁是个过程。

Bean的生命周期


20210613191045315.png
  • 在@ComponentScan注解里对配置项lazyInit配置true可以对Bean的初始化延迟,即Spring不会在发布Bean定义后马上进行实例化与依赖注入;

    • @ComponentScan(lazyInit = true)
  • 可以通过实现上图的接口,进行自定义Bean的生命周期各项属性;
  • 注解@PostConstruct定义了初始化方法,注解@PreDestroy定义了销毁方法;


6. Bean的属性配置

这里指如何给Bean的属性赋值

@Value

  • 属性值
  • 常用于属性与方法上;
  • 可以使用Spring EL表达式对需要配置的属性进行运算;

    • ${......}:表示占位符,读取上下文(application配置文件)的属性值进行装配;
    • #{......}:表示启用Spring表达式,具有运算功能;
    • T(......):表示引入类;
  • 表示给Bean赋值,可以用${xxx}占位符读取application配置文件的内容;

    //获取application里的database.driverName值装配进方法里
    @Value("${database.driverName}")
    private String driverName;
    
    //赋值字符串,同理可以赋值整形、浮点数等
    @Value("#{'赋值字符串'}")
    private String str;
    
    //str后跟着?,表示判断str是否为空,不为空才执行toUpperCase方法
    @Value("#{beanName.str?.toUpperCase()}")
    private String otherBeanProp;
    
    //字符串比较
    @Value("#{beanName.str eq 'SpringBoot'}")
    private boolean strFlag;
    
    //调用方法,System是Java默认加载的包,不用写全类名;其他包要
    @Value("#{T(System).currentTimeMillis()}")
    private Long initTime;

@ConfigurationProperties

  • 配置属性
  • 常标注在类上;
  • 解决@Value过多问题;

    @ConfigurationProperties(“database”)
    public class DataBaseProperties{
    }

@PropertySource

  • 配置来源
  • 添加在主程序类上;
  • 当配置文件不在properties时,例jdbc.properties,需要在主程序类上标注该注解;
  • value可配置多个文件,使用classpath前缀说明在类路径下找;
  • ignoreResourceNotFound默认值false,表示没要找到配置文件就报错。

    • @PropertySource(value={"classpath:jdbc.properties"}, ignoreResourceNotFound=true)


7. Bean的作用域

在Bean的作用域里,常用的是以下加粗的四种。前面IoC容器里讨论的isSingleton与isPrototype方法本质上是作用域的问题。

20210613200200858.png
  • 对于application域而言,完全可以使用单例替代;


@Scope

  • 作用域
  • 常加在类上;
  • 用于修改作用域;
  • ConfigurableBeanFactory仅提供SCOPE_PROTOTYPE (原型)和SCOPE_SINGLETON(单例)两种;
  • 在SpringMVC环境中,WebApplicationContext提供SCOPE_REQUEST(请求)、SCOPE_SESSION(会话)、SCOPE_APPLICATION(应用)三种。

    @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    public class ScopeBean{
    }
    //在同一个请求范围内去获取Bean时,只会共用一个Bean,第二次请求会产生新的Bean
    @Scope(WebApplicationContext.SCOPE_REQUEST) 
    public class ScopeBean{
    }

8. 环境切换

在实际开发中,项目往往需要在开发环境、测试环境、准生产环境和生产环境中切换,Spring提供了Profile机制,方便在各个环境间切换。

@Profile

  • 配置切换
  • 常用于不同环境下需要不用配置的方法或类上;
  • 可以在application中配置两个参数修改Profile机制,spring.profiles.activespring.profiles.default,当这两个属性都没配置情况下,Spring不会启动Profile机制。其中,前者优先级高。

    • spring.profiles.active=dev

      @Profile("dev")
      public DataSource getDataSource( ) {
      }
      
      @Profile("dev")
      public DataSource getDataSource( ) {
      }


# 9. 引入XML文件

虽然SpringBoot不建议使用XML来进行属性配置,但在某些情况下我们又不得不使用XML来进行配置,如:Dubbo框架基于Spring的XML方式进行开发。需要将其引入才能作用。

@ImportResources

  • 导入资源
  • 常用于需要XML装配是配置类上;

    @ImportResources(value = {" classpath:spring-other.xml"})
    public class AppConfig{
    }
相关文章
|
15天前
|
XML Java 测试技术
Spring IOC—基于注解配置和管理Bean 万字详解(通俗易懂)
Spring 第三节 IOC——基于注解配置和管理Bean 万字详解!
99 26
|
18天前
|
缓存 Java 数据库
SpringBoot缓存注解使用
Spring Boot 提供了一套方便的缓存注解,用于简化缓存管理。通过 `@Cacheable`、`@CachePut`、`@CacheEvict` 和 `@Caching` 等注解,开发者可以轻松地实现方法级别的缓存操作,从而提升应用的性能和响应速度。合理使用这些注解可以大大减少数据库的访问频率,优化系统性能。
165 89
|
5天前
|
监控 Java Spring
SpringBoot:SpringBoot通过注解监测Controller接口
本文详细介绍了如何通过Spring Boot注解监测Controller接口,包括自定义注解、AOP切面的创建和使用以及具体的示例代码。通过这种方式,可以方便地在Controller方法执行前后添加日志记录、性能监控和异常处理逻辑,而无需修改方法本身的代码。这种方法不仅提高了代码的可维护性,还增强了系统的监控能力。希望本文能帮助您更好地理解和应用Spring Boot中的注解监测技术。
33 16
|
2月前
|
XML Java 数据格式
【SpringFramework】Spring IoC-基于XML的实现
本文主要讲解SpringFramework中IoC和DI相关概念,及基于XML的实现方式。
116 69
|
22天前
|
监控 Java 应用服务中间件
SpringBoot是如何简化Spring开发的,以及SpringBoot的特性以及源码分析
Spring Boot 通过简化配置、自动配置和嵌入式服务器等特性,大大简化了 Spring 应用的开发过程。它通过提供一系列 `starter` 依赖和开箱即用的默认配置,使开发者能够更专注于业务逻辑而非繁琐的配置。Spring Boot 的自动配置机制和强大的 Actuator 功能进一步提升了开发效率和应用的可维护性。通过对其源码的分析,可以更深入地理解其内部工作机制,从而更好地利用其特性进行开发。
42 6
|
1月前
|
缓存 安全 Java
Spring Boot 3 集成 Spring Security + JWT
本文详细介绍了如何使用Spring Boot 3和Spring Security集成JWT,实现前后端分离的安全认证概述了从入门到引入数据库,再到使用JWT的完整流程。列举了项目中用到的关键依赖,如MyBatis-Plus、Hutool等。简要提及了系统配置表、部门表、字典表等表结构。使用Hutool-jwt工具类进行JWT校验。配置忽略路径、禁用CSRF、添加JWT校验过滤器等。实现登录接口,返回token等信息。
348 12
|
2月前
|
Java Spring 容器
【SpringFramework】Spring IoC-基于注解的实现
本文主要记录基于Spring注解实现IoC容器和DI相关知识。
60 21
|
2月前
|
设计模式 XML Java
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
本文详细介绍了Spring框架的核心功能,并通过手写自定义Spring框架的方式,深入理解了Spring的IOC(控制反转)和DI(依赖注入)功能,并且学会实际运用设计模式到真实开发中。
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
|
1月前
|
存储 安全 Java
Spring Boot 3 集成Spring AOP实现系统日志记录
本文介绍了如何在Spring Boot 3中集成Spring AOP实现系统日志记录功能。通过定义`SysLog`注解和配置相应的AOP切面,可以在方法执行前后自动记录日志信息,包括操作的开始时间、结束时间、请求参数、返回结果、异常信息等,并将这些信息保存到数据库中。此外,还使用了`ThreadLocal`变量来存储每个线程独立的日志数据,确保线程安全。文中还展示了项目实战中的部分代码片段,以及基于Spring Boot 3 + Vue 3构建的快速开发框架的简介与内置功能列表。此框架结合了当前主流技术栈,提供了用户管理、权限控制、接口文档自动生成等多项实用特性。
82 8
|
2月前
|
存储 Java 应用服务中间件
【Spring】IoC和DI,控制反转,Bean对象的获取方式
IoC,DI,控制反转容器,Bean的基本常识,类注解@Controller,获取Bean对象的常用三种方式