前面两篇学习了xml注入和java注入bean到Spring容器,但这两种都需要一个个类去配置文件中编写,未免太过麻烦,这篇,我们就了解一下通过包扫描的方式进行注入!
重点标识
构造器注入,@Autowired注入,以及@Resource注入的区别,官方为何推荐构造器注入,它的好处与坏处。
准备代码
这里提前说明一下,实际上@Controller,@Repository,@Service甚至是@ComponentScan(basePackages = “org.tongzhou”)从技术上来看,本质上是一样的,都是往Spring容器中去注入Bean,只是为了业务的区别,所以用了不同的名称。
我们点进去注解,可以发现,本质上他的内部,几乎没什么区别,如图:
好了,言归正传,看看代码示例:
Dao层
@Repository
public class UserDao {
public void say(){
System.out.println("hello……");
}
}
Service层,这里我采用了@Autowired注入,有人说@Autowired默认是通过类型来查找的,如果找不到,就会报错,实际上这句话还少了一部分,@Autowired确实是通过类型来查找的,但是找不到,他会再根据名称去查找,如果名称也找不到,才会报错。
@Service
public class UserService {
@Autowired
UserDao userDao;
public void say(){
userDao.say();
}
}
Controller层,这里呢,为了简单就不写接口了,我怕们直接从Spring容器中拿,重点看一下,这边我又改成了构造器注入,大家知道,类在加载的时候,是会去加载构造器的,在这个示例中,Spring在加载UserController 的时候,就会发现userService,顺带着也找到它。
@Controller
public class UserController {
UserService userService;
public void say(){
userService.say();
}
public UserController(UserService userService) {
this.userService = userService;
}
}
最后,别忘记加一个包扫描,先看看java配置的方式,basePackages默认扫描的是当前包下面的所有包里面的类,这样的话,如果层级与他平级的包里面的类就会扫描不到,我们可以指定,如下:
@ComponentScan(basePackages = "org.tongzhou")
public class JavaConfig {
}
再看看xml配置,很简单,一句话:
<context:component-scan base-package="org.tongzhou" />
测试一下看看。
public static void main(String[] args) {
AnnotationConfigApplicationContext act = new AnnotationConfigApplicationContext(JavaConfig.class);
UserController bean = act.getBean(UserController.class);
}
当然是没问题的,
好了,我们现在来看下,为什么官方推荐使用构造器的方式注入,而不推荐@Autowired呢.
在一些特定的情况下,可能你不能从Spring容器中去拿,而是考虑new的方式,这里区别就出来,对于构造器注入的,new一个新对象,一开始,就会提示你,需要注入一个参数,这样就可以避免后续的报错,而对于@Autowired来说,你new出来之后,是没问题的,但是当你运行的时候,就会抱一个找不到类的错误,看看如下演示:
UserService userService = new UserService();
userService.say();
调用后,他就会提示你,找不到,因为对于你这个new出来的新对象根本就没有。
当然,这是一些特殊情况,正常情况下,也没人去new Controller层的东西。
对了,还有一种情况,如果你使用了构造器注入,但是呢,又有多个构造器,这时候,你可以这么做,加上一个@Autowired注解,完美解决这个问题。
@Controller
public class UserController {
UserService userService;
public void say(){
userService.say();
}
@Autowired
public UserController(UserService userService) {
this.userService = userService;
}
public UserController() {
}
}