你可能使用了Spring最不推荐的注解方式

简介: 你可能使用了Spring最不推荐的注解方式

前些天新加入项目中的小伙儿写了一段代码,我看到之后,顿时以固有的经验来判断说:Spring的注解不能这样写,不够简洁云云。但本着对技术的钻研精神,还是深入分析比较了一下Spring的注解形式及优缺点。原来,大家最常用的注解方式不是最优的,而是最不推荐的。

前言


使用Spring框架最核心的两个功能就是IOC和AOP。IOC也就是控制反转,我们将类的实例化、依赖关系等都交由Spring来处理,以达到解耦合、利用复用、利于测试、设计出更优良程序的目的。而对用户来说,操作最对的便是注解。在Spring中提供了三类注解方式,下面我们就逐一分析。最后,你会发现,你最常用、看起来最方便的形式确实最不推荐的一种形式。

常见的注入方式


Field注入



  1. @Controller
  2. publicclassFooController{
  3.  @Autowired
  4.  // @Resource
  5.  privateFooService fooService;
  6. }

此种注解方式,应用最广泛:

  • 注入简单,只需在字段上添加@Autowired或@Resource;
  • 减少大量冗余代码,美观;
  • 新增Field时不需要过多代码修改;

构造函数注入


  1. @Controller
  2. publicclassFooController{

  3.  privatefinalFooService fooService;
  4.  privatefinalFooService1 fooService1;

  5.  @Autowired
  6.  publicFooController(FooService fooService,FooService1 fooService1){
  7.      this.fooService = fooService;
  8.      this.fooService1 = fooService1;
  9.  }


  10. }

  1. @Controller
  2. publicclassFooController{

  3.  privatefinalFooService fooService;
  4.  // 当只有一个参数时可不写@Autowired
  5.  publicFooController(FooService fooService){
  6.      this.fooService = fooService;
  7.  }
  8. }

Spring4.x推荐的注入方式。对比Field注入:

  • 代码臃肿
  • 新增Field修改麻烦
  • 当Field多余5个时不符合构造方法的基本规范,显得笨重、臃肿;

setter注入



  1. @Controller
  2. publicclassFooController{

  3.  privateFooService fooService;

  4.  @Autowired
  5.  publicvoid setFooService(FooService fooService){
  6.      this.fooService = fooService;
  7.  }
  8. }

Spring3.x推荐的注入方式,但并没有被广泛应用,当初推荐的理由:

  • 解决了构造器注入的笨重;
  • 可以让类在之后重新配置或者重新注入。


为什么Spring4.x推荐构造函数注入




在上面的分析看来,构造函数注入好像并没有显现出来它的优势,但问什么Spring4.x会推翻之前推荐的setter注入,采用构造函数注入呢?官方的理由汇总如下:

  • 依赖不可变:加入了final来约束修饰Field,这条是很显然的;
  • 依赖不可为空:在实例化的时候会检查构造函数参数是否为空,如果为空(未找到改类型的实例对象)则会抛出异常。
  • 单一职责:当使用构造函数注入时,如果参数过多,你会发现当前类的职责过大,需要进行拆分。而使用Field注入时,你并不会意识到此问题。
  • 更利于单元测试:按照其他两种方式注入,当单元测试时需要初始化整个spring的环境,而采用构造方法注入时,只需要初始化需要的类即可,即可以直接实例化需要的类。
  • 避免IOC容器以外环境调用时潜在的NPE(空指针异常)。
  • 避免循环依赖。
  • 保证返回客户端(调用)的代码的时候是完全初始化的状态。


疑问



如果有大量依赖需要注入,怎么办?



如果有大量依赖需要注入说明该类的职责过于复杂,需要遵从单一性原则进行拆分;


其他注入方式是否合理



存在即合理,根据具体情况可以采用最适合的方式。比如,可以同时使用@Qualifier来达到一些约束限定的目的。也可以使用setter注入和构造函数注入相结合的方式来进行注入。

Idea插件支持


Idea已经提供了相应的插件支持:9.gif

目录
相关文章
|
11天前
|
XML Java 数据格式
SpringBoot入门(8) - 开发中还有哪些常用注解
SpringBoot入门(8) - 开发中还有哪些常用注解
29 0
|
1月前
|
Java Spring 容器
如何解决spring EL注解@Value获取值为null的问题
本文探讨了在使用Spring框架时,如何避免`@Value("${xxx.xxx}")`注解导致值为null的问题。通过具体示例分析了几种常见错误场景,包括类未交给Spring管理、字段被`static`或`final`修饰以及通过`new`而非依赖注入创建对象等,提出了相应的解决方案,并强调了理解框架原理的重要性。
135 4
|
29天前
|
Java Spring
在使用Spring的`@Value`注解注入属性值时,有一些特殊字符需要注意
【10月更文挑战第9天】在使用Spring的`@Value`注解注入属性值时,需注意一些特殊字符的正确处理方法,包括空格、引号、反斜杠、新行、制表符、逗号、大括号、$、百分号及其他特殊字符。通过适当包裹或转义,确保这些字符能被正确解析和注入。
|
18天前
|
XML JSON Java
SpringBoot必须掌握的常用注解!
SpringBoot必须掌握的常用注解!
41 4
SpringBoot必须掌握的常用注解!
|
1月前
|
XML Java 数据格式
Spring从入门到入土(bean的一些子标签及注解的使用)
本文详细介绍了Spring框架中Bean的创建和使用,包括使用XML配置文件中的标签和注解来创建和管理Bean,以及如何通过构造器、Setter方法和属性注入来配置Bean。
67 9
Spring从入门到入土(bean的一些子标签及注解的使用)
|
19天前
|
存储 缓存 Java
Spring缓存注解【@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig】使用及注意事项
Spring缓存注解【@Cacheable、@CachePut、@CacheEvict、@Caching、@CacheConfig】使用及注意事项
61 2
|
19天前
|
JSON Java 数据库
SpringBoot项目使用AOP及自定义注解保存操作日志
SpringBoot项目使用AOP及自定义注解保存操作日志
34 1
|
1月前
|
架构师 Java 开发者
得物面试:Springboot自动装配机制是什么?如何控制一个bean 是否加载,使用什么注解?
在40岁老架构师尼恩的读者交流群中,近期多位读者成功获得了知名互联网企业的面试机会,如得物、阿里、滴滴等。然而,面对“Spring Boot自动装配机制”等核心面试题,部分读者因准备不足而未能顺利通过。为此,尼恩团队将系统化梳理和总结这一主题,帮助大家全面提升技术水平,让面试官“爱到不能自已”。
得物面试:Springboot自动装配机制是什么?如何控制一个bean 是否加载,使用什么注解?
|
14天前
|
存储 安全 Java
springboot当中ConfigurationProperties注解作用跟数据库存入有啥区别
`@ConfigurationProperties`注解和数据库存储配置信息各有优劣,适用于不同的应用场景。`@ConfigurationProperties`提供了类型安全和模块化的配置管理方式,适合静态和简单配置。而数据库存储配置信息提供了动态更新和集中管理的能力,适合需要频繁变化和集中管理的配置需求。在实际项目中,可以根据具体需求选择合适的配置管理方式,或者结合使用这两种方式,实现灵活高效的配置管理。
11 0
|
1月前
|
XML Java 数据库
Spring boot的最全注解
Spring boot的最全注解