创建bean的过程:
1.使用类的构造方法实例化得到一个普通对象(对于选择构造方法的过程称为推断构造方法,如果是有一个构造方法直接使用;如果有多个构造方法,查看是否存在无参的构造方法,存在则直接使用;不存在的话,如果某个构造方法上有@Autowired注解则选择此构造方法,如果都没有则会抛出异常,初始化bean失败)
2.得到一个对象后,spring会判断当前类中是否存在@Autowired注解了的属性,把这些属性进行赋值(此过程称为依赖注入)
3.在依赖注入之后,spring会判断当前类是否实现了BeanNameAware接口,BeanClassLoaderAware接口,BeanFactoryAware接口,如果实现了,表示当前对象必须实现这些接口定义的setBeanName(),setBeanClassLoader(),setBeanFactory()方法,spring调用这些方法并且传入相应的参数(称为Aware回调)。
4.在Aware回调之后,spring需要判断当前对象的类中是否存在某些方法上使用了@PostConstruct注解,存在则spring需要调用这些方法。(初始化前)
5.之后,spring会判断当前对象对应的类是否实现了InitializingBean,如果实现了则当前对象必须实现afterPropertiesSet()方法,spring就会调用当前对象的afterPropertiesSet()方法。(初始化)
6.最后,spring会判断当前对象是否需要Aop,需要的话,使用动态代理从而生成一个代理对象作为Bean;不需要的话,bean就创建完成。
扩展:
1.为什么同一个类中两个方法a,b都加上了事务的注解,然后在a中调用b,事务不会生效?
因为事务对象是一个代理对象,当代理对象调用方法仅调用a时,事务是有效的,此时使用的是代理对象;如果调用了方法a,再在a中调用b时,此时的对象已经被赋值为普通对象了,普通对象调用b,普通对象不是事务的代理对象,无法生效。
解决方案:1.把两个方法分别写在不同的类中(经常使用)
2.在当前方法中自己注入自己,即增加UserService属性注入,使用userService.b()方法调用即可生效。