还在用 @Autowired 和 @Resource 方式实现依赖注入吗?
之前接了一个项目里看到有的同事用到了@RequiredArgsConstructor
,省略了@Autowired/@Resource
,于是就带着好奇去研究了一下。
简介
RequiredArgsConstructor
是 Lombok
提供的一个注解,它可以自动生成构造函数,该构造函数包含所有标记为final
或@NonNull
的字段,这个注解可以减少代码量,提高代码的可读性和可维护性。
先回顾一下依赖注入常见的几种实现方式:
- set注入
- 构造器注入
- 属性注入
- 形参上注入
基础类
@Service
public class UserServiceImpl implements UserService {
@Override
public void out() {
System.out.println("Service层执行结束");
}
}
set注入
@Controller
public class UserController {
private UserService userService;
@Autowired
public void setUserService(UserService userService) {
this.userService = userService;
}
public void out() {
userService.out();
System.out.println("Controller层执行结束。");
}
}
构造器注入
@Controller
public class UserController {
private UserService userService;
@Autowired
public UserController(UserService userService) {
this.userService = userService;
}
public void out() {
userService.out();
System.out.println("Controller层执行结束。");
}
}
属性注入
@Controller
public class UserController {
@Autowired
private UserService userService;
public void out() {
userService.out();
System.out.println("Controller层执行结束。");
}
}
形参上注入
@Controller
public class UserController {
private UserService userService;
public UserController(@Autowired UserService userService) {
this.userService = userService;
}
public void out() {
userService.out();
System.out.println("Controller层执行结束。");
}
}
总结:
@Autowired
注解可以出现在:属性上、构造方法上、构造方法的参数上、setter
方法上。- 当带参数的构造方法只有一个,
@Autowired
注解可以省略。@Autowired
注解默认根据类型注入。如果要根据名称注入的话,需要配合@Qualifier
注解一起使用扩展说明(
Java
初始化类的顺序):父类的静态字段 > 父类静态代码块 > 子类静态字段 > 子类静态代码块 > 父类成员变量 > 父类构造代码块 > 父类构造器 > 子类成员变量 > 子类构造代码块 > 子类构造器
而
Autowired
注入,则要排队到子类构造器以后了,SpringIOC
并不会对依赖的bean
是否为null
做判断,JVM编译时同样也不会有问题,但如果使用不当,运行起来时或许会因为出现空指针异常,这也许是@Autowired
为什么会被IDEA
警告的原因之一吧
这里有必要提一下@Resource
:
@Resource
注解是JDK扩展包中的,也就是说属于JDK
的一部分。所以该注解是标准注解,更加具有通用性。(JSR-250
标准中制定的注解类型。JSR
是Java
规范提案。),@Autowired
注解是Spring
框架自己的@Resource
注解默认根据名称装配byName
,未指定name
时,使用属性名作为name
。通过name
找不到的话会自动启动通过类型byType
装配。@Resource
注解用在属性上、setter
方法上。
正文
引入依赖:
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
</dependency>
使用说明
类上加上
@RequiredArgsConstructor
,需要注入的类要用final
声明,或者使用@NonNull
。
@Controller
@RequiredArgsConstructor(onConstructor_ = {@Lazy, @Autowired})
public class UserController {
private final UserService userService;
public void out() {
userService.out();
System.out.println("Controller层执行结束。");
}
}
试想实际开发中我们会有Service
或Dao
需要注入,这样就要写上好多@Autowired/@Resource
,而@RequiredArgsConstructor(onConstructor_ = {@Lazy, @Autowired})
一行帮你搞定,是不是很方便。
细心的你可能发现小编这里不是直接使用的@RequiredArgsConstructor
,而是增加了(onConstructor_ = {@Lazy, @Autowired})
,为什么呢?
@Component
@RequiredArgsConstructor
public class A {
@NonNull
private B b;
}
@Component
@RequiredArgsConstructor
public class B {
private final A a;
}
启动程序,出现异常,异常意思是说出现了循环依赖:
Description:
The dependencies of some of the beans in the application context form a cycle:
┌─────┐
| a defined in file [D:\local\inetAddress\target\classes\org\example\test\A.class]
↑ ↓
| b defined in file [D:\local\inetAddress\target\classes\org\example\test\B.class]
└─────┘
解决方案:
- 代码重构
@Lazy
注解- 属性注入
所以小编直接加了(onConstructor_ = {@Lazy, @Autowired})
,
onConstructor_参数 :
- onConstructor:列出的所有注解都放在生成的构造方法上
JDK 7 之前的写法是 onConstructor = @__({@Deprecated}),而 JDK 8 之后的写法是 onConstructor_ = {@Deprecated}