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注入的方式来指定注入类型。

posted @ 2024-02-16 14:48  leviH  阅读(178)  评论(0编辑  收藏  举报