大家好,我是一安~
之前接了一个项目里看到有的同事用到了@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:localinetAddresstargetclassesorgexampletestA.class]
↑ ↓
| b defined in file [D:localinetAddresstargetclassesorgexampletestB.class]
└─────┘
解决方案:
-
代码重构 -
@Lazy
注解 -
属性注入
所以小编直接加了(onConstructor_ = {@Lazy, @Autowired})
,至于循环依赖更详细的本篇不做介绍,有兴趣可以看之前文章《大厂Java二面:Spring是如何解决循环依赖的问题》
如果这篇文章对你有所帮助,或者有所启发的话,帮忙 分享、收藏、点赞、在看,你的支持就是我坚持下去的最大动力!
原文始发于微信公众号(一安未来):还在用 @Autowired 和 @Resource 方式实现依赖注入吗?
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/145093.html