你好,我是柳岸花开。
在Spring框架中,依赖注入(Dependency Injection)是一种重要的实现方式,它通过将对象依赖关系的管理交给Spring容器来完成,使得代码更加灵活、可维护性更高。在Spring中,有多种注入方式,本文将详细介绍手动注入和自动注入两种主要方式,并对寻找注入点和注入对象进行深入分析。
注入方式
手动注入
手动注入是通过配置XML或者Java代码来完成依赖关系的管理。在Spring中,手动注入有两种主要方式:set方式注入和构造方法注入。
set方式注入
通过XML配置文件,使用<property>
标签为Bean的属性赋值,示例如下:
<bean name="userService" class="com.bob.service.UserService">
<property name="orderService" ref="orderService"/>
</bean>
或者使用构造方法注入:
构造方法注入
<bean name="userService" class="com.bob.service.UserService">
<constructor-arg index="0" ref="orderService"/>
</bean>
自动注入
自动注入是Spring框架中更为便捷的注入方式,它不需要手动配置每个Bean的依赖关系,而是由Spring框架自动完成。自动注入包括XML的autowire自动注入和@Autowired注解的自动注入。
XML的autowire自动注入
通过在XML配置文件中设置autowire
属性,可以实现自动注入。例如:
<bean id="userService" class="com.bob.service.UserService" autowire="byType"/>
Spring会自动为userService
中的所有属性自动赋值,底层实现是基于set方法注入和构造方法注入。
自动注入模式包括:
-
byType
: 根据属性的类型自动装配,如果找到多个符合类型的Bean则报错。 -
byName
: 根据属性的名称自动装配,通过名称匹配Bean。 -
constructor
: 通过构造方法注入,可以不写set方法。 -
default
/no
: 关闭自动注入。
@Autowired注解的自动注入
@Autowired注解是一种更为灵活的自动注入方式,它可以用于属性、构造方法、或者set方法上。
@Component
public class Example {
@Autowired
private SomeService someService;
@Autowired
public Example(SomeService someService) {
this.someService = someService;
}
}
@Autowired注解提供了细粒度的控制,能够指定具体的注入方式以及要注入的Bean。
寻找注入点
在创建一个Bean的过程中,Spring会利用AutowiredAnnotationBeanPostProcessor的postProcessMergedBeanDefinition()找出注入点并缓存。
找注入点的流程为:
-
遍历当前类的所有的属性字段Field -
查看字段上是否存在@Autowired、@Value、@Inject中的其中任意一个,存在则认为该字段是一个注入点 -
如果字段是static的,则不进行注入 -
获取@Autowired中的required属性的值 -
将字段信息构造成一个AutowiredFieldElement对象,作为一个注入点对象添加到currElements集合中。 -
遍历当前类的所有方法Method -
判断当前Method是否是桥接方法,如果是找到原方法 -
查看方法上是否存在@Autowired、@Value、@Inject中的其中任意一个,存在则认为该方法是一个注入点 -
如果方法是static的,则不进行注入 -
获取@Autowired中的required属性的值 -
将方法信息构造成一个AutowiredMethodElement对象,作为一个注入点对象添加到currElements集合中 -
遍历完当前类的字段和方法后,将遍历父类的,直到没有父类 -
最后将currElements集合封装成一个InjectionMetadata对象,作为当前Bean对于的注入点集合对象,并缓存
注入点进行注入
Spring在AutowiredAnnotationBeanPostProcessor的postProcessProperties()方法中,会遍历所找到的注入点依次进行注入。
字段注入:
-
遍历所有的AutowiredFieldElement对象。 -
将对应的字段封装为DependencyDescriptor对象。 -
调用BeanFactory的resolveDependency()方法,传入DependencyDescriptor对象,进行依赖查找,找到当前字段所匹配的Bean对象。 -
将DependencyDescriptor对象和所找到的结果对象beanName封装成一个ShortcutDependencyDescriptor对象作为缓存,比如如果当前Bean是原型Bean,那么下次再来创建该Bean时,就可以直接拿缓存的结果对象beanName去BeanFactory中去那bean对象了,不用再次进行查找了 -
利用反射将结果对象赋值给字段。
Set方法注入:
-
遍历所有的AutowiredMethodElement对象 -
遍历将对应的方法的参数,将每个参数封装成MethodParameter对象 -
将MethodParameter对象封装为DependencyDescriptor对象 -
调用BeanFactory的resolveDependency()方法,传入DependencyDescriptor对象,进行依赖查找,找到当前方法参数所匹配的Bean对象。 -
将DependencyDescriptor对象和所找到的结果对象beanName封装成一个ShortcutDependencyDescriptor对象作为缓存,比如如果当前Bean是原型Bean,那么下次再来创建该Bean时,就可以直接拿缓存的结果对象beanName去BeanFactory中去那bean对象了,不用再次进行查找了 -
利用反射将找到的所有结果对象传给当前方法,并执行。
获取Bean对象
public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory
implements ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable {
// 根据DependencyDescriptor依赖描述从BeanFactory中找出对应的唯一的一个Bean对象。
public Object resolveDependency(DependencyDescriptor descriptor, @Nullable String requestingBeanName,
@Nullable Set<String> autowiredBeanNames, @Nullable TypeConverter typeConverter) throws BeansException {
}
}
resolveDependency()
-
判断descriptor.getDependencyType()
-
如果是Optional.class,返回包装成Optional的对象 -
ObjectFactory或ObjectProvider,返回DependencyObjectProvider对象 -
如果是javax.inject.Provider类对象,则返回Jsr330Provider对象 -
如果依赖描述上有@Lazy注解,那么会生成一个代理对象然后返回,那么在调用代理对象中的方法时调用doResolveDependency(descriptor)。或这,直接生成一个ObjectFactory对象,在调用ObjectFactory的getObject方法时调用调用doResolveDependency(descriptor)
-
doResolveDependency()方法
-
是否存在@Value注解,如果存在,获取并解析descriptor上的@Value注解,进行解析并返回 -
如果不存在@Value注解,判断descriptor的类型
1.如果是Map,直接返回map
2.如果是Collection,则会调用方法findAutowireCandidates(beanName, type, descriptor),该方法返回一个Map,表示会根据type去找bean,Map的key为beanName,Map的value为对象(注意可能是bean对象,也可能是某个bean的class对象,因为该方法只负责根据类型找到对应的bean,如果该bean还没有实例化,那么该方法不负责去实例化,只返回该Bean对应的Class对象,表示这个Bean也是结果之一)
本文由 mdnice 多平台发布