Spring中到底有几种依赖注入的方式
首先分两种:
- 手动注入
- 自动注入
手动注入
在XML中定义Bean时,就是手动注入,因为是
程序员手动给某个属性指定了值
。
<bean name="userService" class="com.luban.service.UserService">
<property name="orderService" ref="orderService"/>
</bean>
上面这种底层是通过
set方法
进行注入。
<bean name="userService" class="com.luban.service.UserService">
<constructor‐arg index="0" ref="orderService"/>
</bean>
上面这种底层是通过
构造方法
进行注入。
所以手动注入的底层也就是分为两种:
1. set方法注入
2. 构造方法注入
自动注入
自动注入又分为两种:
1. spring自带的autowire自动注入
2. @Autowired、@Resource、@Value、@Inject注解的自动注入
spring自带的autowire自动注入
在XML或@bean中,我们可以在定义一个Bean时去指定这个Bean的自动注入模式,这种方式会忽略简单字段类型的注入,如String、Number:
- byType
- byName
- constructor
- default
- no
比如:
<bean id="userService" class="com.luban.service.UserService" autowire="byType"/>
这么写,表示Spring会自动的给userService中所有的set方法注入属性。set方法不一定要有对象的属性。
在创建Bean的过程中,在填充属性时,Spring会去解析当前类,把
当前类的所有方法
都解析出来,
Spring会去解析每个方法得到对应的PropertyDescriptor对象,PropertyDescriptor中有几个属性:
1.
name:这个name并不是方法的名字,而是拿方法名字进过处理后的名字
i.
如果方法名字以“get”开头,比如“getXXX”,那么name=XXX
ii.
如果方法名字以“
is”开头,比如“
isXXX”,那么name=XXX
iii.
如果方法名字以“
set”开头,比如“
setXXX”,那么name=XXX
2.
readMethodRef:表示get方法的Method对象的引用
3.
readMethodName:表示get方法的名字
4.
writeMethodRef:表示set方法的Method对象的引用
5.
writeMethodName:表示set方法的名字
6.
propertyTypeRef:如果有get方法那么对应的就是返回值的类型,如果是set方法那么对应的
就是set方法中唯一参数的类型
get方法的定义是:
方法参数个数为0个,并且 (方法名字以"get"开头 或者 方法名字以"is"开头并
且方法的返回类型为boolean)
set方法的定义是:**方法参数个数为1个,并且 (方法名字以"set"开头并且方法返回类型为
void)
所以,Spring在通过byName的自动填充属性时流程是:
1. 找到所有set方法所对应的XXX部分的名字
2. 根据XXX部分的名字去获取bean
Spring在通过byType的自动填充属性时流程是:
1. 获取到set方法中的唯一参数的参数类型,并且根据该类型去容器中获取bean
2. 如果找到多个,会报错。
如果是constructor,那么就可以不写set方法了,当某个bean是通过构造方法来注入时,spring利用 构造方法的参数信息从Spring容器中去找bean,找到bean之后作为参数传给构造方法,从而实例化 得到一个bean对象,并完成属性赋值(属性赋值的代码得程序员来写)。
其实构造方法注入相当于byType+byName,普通的byType是根据set方法中的参数类型去找bean,找到多个会报错,而constructor就是通过构造方法中的参数类型去找bean,如果找到多个会根据参数名确定。
@Autowired注解的自动注入
@Autowired注解,是byType和byName的结合。
@Autowired注解可以写在:
1. 属性上:先根据
属性类型
去找Bean,如果找到多个再根据
属性名
确定一个
2. 构造方法上:先根据方法
参数类型
去找Bean,如果找到多个再根据
参数名
确定一个
3. set方法上:先根据方法
参数类型
去找Bean,如果找到多个再根据
参数名
确定一个
这个注解一般和@Qualifier配合使用,如果根据属性名称找不到bean,再根据@Qualifier的值找。
@Inject注解的自动注入
这个注解注入的原理与@Autowired一致,只是它没有属性,不能设置为requied=true/false。
@Resource注解的自动注入
如果
@Resource指定了name属性,则根据name值为name找bean,找不到直接报错。如果没指定,那么先根据属性名找,找不到再根据属性类型找,如果找到多个,再根据属性名称判断,如果找不到,则报错。@Autowide是spring 的注解,而@Resource是java自带的注解。