Spring常见三种注入方式
1、field注入(字段注入)
1 @Controller 2 public class FooController { 3 @Autowired 4 private FooService fooService; 5 6 //简单的使用例子,下同 7 public List<Foo> listFoo() { 8 return fooService.list(); 9 } 10 }
最常见的注入方式、注入方式简单明了,类中定义全局变量,可以使用@Autowired、@Resource、@Inject注解来实现
2、构造器注入
1 @Controller 2 public class FooController { 3 4 private final FooService fooService; 5 6 public FooController(FooService fooService) { 7 this.fooService = fooService; 8 } 9 //使用方式上同 10 public List<Foo> listFoo() { 11 return fooService.list(); 12 } 13 }
定义final类型变量,并定义一个带有带变量的构造函数,在构造函数初始化该变量,变量不能是static类型。Spring4.x版本中推荐的注入方式。如果在需要注入的变量很多的情况下,构造很熟会很大很影响代码美观,所以推荐lombok的注解@RequiredArgsConstructor,该注解作用于类上,可以给当前生成一个构造函数,该构造函数的参数为当前类中被@NotNull注解修饰的变量和final类型的变量。
3、setter注入
1 @Controller 2 public class FooController { 3 private FooService fooService; 4 5 @Autowired 6 public void setFooService(FooService fooService) { 7 this.fooService = fooService; 8 } 9 10 //使用方式上同 11 public List<Foo> listFoo() { 12 return fooService.list(); 13 } 14 }
定义一个变量,并实现该变量的set方法,set方法需要加上注解@Autowired,Spring3.x刚推出的时候,推荐使用注入的就是这种。改注解的优点就是可以在类使用之后重新配置或者重新注入。
构造器注入的好处:
构造器注入是Spring4.x版本中推荐的注入方式,Spring官方对此也有解释,翻译过来就是构造器注入的方式啊,能够保证注入的**依赖不可变**,并且确保需要的**依赖不为空**。此外,构造器注入的依赖总是能够在返回客户端(组件)代码的时候保证**完全初始化的状态**。
1、依赖不可变:就是变量上final关键字的作用。
2、依赖不为空:因为是通过构造函数注入的,Spring容器在实例化的时候会调用构造函数传入具体值。
3、完全初始化的状态:这个可以跟上面的依赖不为空结合起来,向构造器传参之前,要确保注入的内容不为空,那么肯定要调用依赖组件的构造方法完成实例化。而在Java类加载实例化的过程中,构造方法是最后一步(之前如果有父类先初始化父类,然后自己的成员变量,最后才是构造方法)。所以返回来的都是初始化之后的状态。
field注入的缺点:
1、对于IOC容器之外的环境,会报错NPE
1 FooController fooController = new FooController(); 2 fooController.listFoo(); //NullPointerException
2、在循环依赖的场景下,构造器注入在项目启动启动时会报错、而field注入的方式启动时不会报错,在使用具体的对象时才会报错
另外,在一个依赖的接口有多个实现类时,推荐使用field注入或者setter注入的方式来指定注入类型。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· winform 绘制太阳,地球,月球 运作规律
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人