先说结论: 实际开发时尽量选择 构造器注入 而不是 字段注入。 设值注入 和 方法注入 适用的场景较少,较特殊。
原因:
● 官方推荐使用!
● 代码更简洁:配合 Lombok 的 @RequiredArgsConstructor,可以自动生成包含所有 final 字段的构造函数。
● 更安全:构造器注入确保所有必需的依赖在对象创建时就已经注入,避免了在运行时忘记注入某个依赖导致 NullPointerException 的问题。
● 更易于测试:使用构造器注入时,可以直接通过构造器传递依赖进行单元测试,而不需要模拟框架的注入机制。
至于 @Resource它是 Java 的 javax.annotation.Resource 注解,个人建议在Spring项目中就不要用了。
两者区别如下:
@Autowired
● Spring 提供的注解:@Autowired 是 Spring 容器提供的注解,专门用于依赖注入。
● 默认按类型注入:@Autowired 默认按照类型(type)进行注入。
● 可指定按名称注入:可以通过 @Qualifier 注解指定按名称注入。
@Resource
● J2EE 标准注解:@Resource 是 J2EE 标准的一部分,通常用于 EJB 和其他 J2EE 资源的注入。
● 默认按名称注入:@Resource 默认按照名称(name)进行注入,如果找不到匹配的名称,则按类型(type)注入。
● 不需要 @Qualifier:@Resource 可以通过 name 属性指定名称,而不需要额外的 @Qualifier 注解。
一、构造器注入(Constructor Injection)
构造器注入通过类的构造方法来传递依赖。
举个例子:
java">@RestController
@RequiredArgsConstructor
public class WarningTaskController {// 注意使用 final 标记不可变private final WarningTaskService taskService;// 多个依赖可以
}
二、设值注入(Setter Injection)
设值注入通过 setter 方法来注入依赖。
举个例子:
使用 Spring Security ,默认内存用户认证的信息可以配置spring.security.user.name=customUserName
就是通过 Setter 注入。源码如下:
java">@ConfigurationProperties(prefix = "spring.security")
public class SecurityProperties {// 省略一些非必要的代码private final User user = new User();public User getUser() {return this.user;}public static class User {private String name = "user";private String password = UUID.randomUUID().toString();private List<String> roles = new ArrayList<>();private boolean passwordGenerated = true;public String getName() {return this.name;}public void setName(String name) {this.name = name;}// 省略其他 getter/setter}}
三、字段注入(Field Injection)
字段注入直接通过字段来注入依赖。
举个例子:
java">@RestController
public class WarningTaskController {// 不推荐使用// 如果依赖非常多,代码会很臃肿@Autowiredprivate final WarningTaskService taskService;}
四、方法注入(Method Injection)
方法注入通过方法参数来注入依赖。
举个例子:
java">@RestController
public class WarningTaskController {private WarningTaskService taskService;// 不推荐 如果对象是可变的@Autowiredpublic void configure(WarningTaskService taskService) {this.taskService = taskService;}// 其他业务逻辑...
}