详解依赖注入与自动装配

在我面试的时候,常会问面试者一个问题,就是依赖注入有几种方式,发现面试者的回答五花八门,有回答两种的,也有回答三种的,四种的。其实正确的答案是两种:构造器注入和setter注入。
提到依赖注入,就不能不说装配。有些初学者老是会把这两个概念搞混,这个博文就是来跟你们讨论这两个概念以及其中详细的原理。
依赖注入的本质就是装配,装配是依赖注入的具体行为。这就是二者的关系。例如:
<bean id="hello" class="com.maven.Hello"><constructor-arg value="hello" /></bean>这是使用构造器注入来装配bean。
<bean id="hello" class="com.maven.Hello" p:hello="hello" />这是使用setter注入,p是spring的名称空间,能够用来代替<property>标签。
以上就是两种依赖注入方式。上面的注入只是基本类型的注入。下面介绍一下经常使用的注入配置:
1.构造器注入对象属性
<bean id="text" class="com.maven.Text" />
<bean id="hello" class="com.maven.Hello"><constructor-arg ref="text" /></bean>
2. 属性注入对象属性
<bean id="hello" class="com.maven.Hello"><property name="text" ref="text" /></bean>
3. 属性为List类型或数组类型属性注入
<bean id="hello" class="com.maven.Hello">
<property name="persons">
<list>
<ref bean="zhangsan" />
<ref bean="zhangsan" />
</list>
</property>
</bean>
list元素的成员也能够是<value>,<bean>,<null />,其中<bean>是用来装配匿名bean的,<null />是用来装配null值的。匿名bean会在下面介绍。
4.属性为set类型的属性注入
set类型与list类型注入是同样的,只是标签改为<set>就能够了。而且里面的元素是不能重复的。
5. 属性为map类型的属性注入
<bean id="hello" class="com.maven.Hello">
<property name="article">
<map>
<entry key="title" value-ref="text" />
<entry key="title" value-ref="text" />
</map>
</property>
</bean>
map类型的键和值能够是任何类型,key-ref用来引用键是bean的,value-ref用来引用值是bean的
6. 属性为Properties类型的属性注入
<bean id="hello" class="com.maven.Hello">
<property name="article">
<props>
<prop key="title">I LOVE YOU</prop>
<prop key="title">I HATE YOU</prop>
</props>
</property>
</bean>
Properties类型的元素为props,每个键值标签是prop,注意要与property标签进行区分。并且Properties类型的键和值必须都是String类型的。而且值是用<prop>标签内容表示的,而不是用value属性。
最后说一下匿名bean,这个比较少用。匿名bean跟匿名内部类是同样的,可是匿名bean不须要实现接口,而且也只能用一次,因此<bean>标签中不用写id属性。例如咱们用匿名bean来做为属性注入时:
<property name="text"><bean class="com.maven.Text" /></property>。匿名bean没有id属性,由于匿名bean只能被使用一次,加上id属性没有意义。
好了,说到这里,你们能够看到用xml装配bean是一件很繁琐的事情,并且咱们还要找到对应类型的bean才能装配。
首先,肯定一下装配的概念。《spring实战》中给装配下了一个定义:建立应用对象之间协做关系的行为称为装配。也就是说当一个对象的属性是另外一个对象时,实例化时,须要为这个对象属性进行实例化。这就是装配。若是一个对象只经过接口来代表依赖关系,那么这种依赖就可以在对象自己绝不知情的状况下,用不一样的具体实现进行切换。可是这样会存在一个问题,在传统的依赖注入配置中,咱们必需要明确要给属性装配哪个bean的引用,一旦bean不少,就很差维护了。基于这样的场景,spring使用注解来进行自动装配,解决这个问题。自动装配就是开发人员没必要知道具体要装配哪一个bean的引用,这个识别的工做会由spring来完成。与自动装配配合的还有“自动检测”,这 个动做会自动识别哪些类须要被配置成bean,进而来进行装配。这样咱们就明白了,自动装配是为了将依赖注入“自动化”的一个简化配置的操做。
装配分为四种:byName, byType, constructor, autodetect。byName就是会将与属性的名字同样的bean进行装配。byType就是将同属性同样类型的bean进行装配。constructor就是经过构造器来将类型与参数相同的bean进行装配。autodetect是constructor与byType的组合,会先进行constructor,若是不成功,再进行byType。具体选择哪种装配方式,须要配置<bean>标签的autowire属性,若是没有配置,默认是byName类型,就是会根据属性的名字来进行自动装配。上面最经常使用的仍是byName和byType。自动装配时,装配的bean必须是惟一与属性进行吻合的,不能多也不能少,有且只有一个能够进行装配的bean,才能自动装配成功。不然会抛出异常。若是要统一全部bean的自动装配类型,能够在<beans>标签中配置default-autowire属性。固然若是配置了autowire属性,咱们依然能够手动装配属性,手动装配会覆盖自动装配。
以上是经过xml配置的方式实现自动装配的,spring2.5以后提供了注解方式的自动装配。可是要使用这些注解,须要在配置文件中配置<context:annotation-config />。只有加上这一配置,才可使用注解进行自动装配,默认状况下基于注解的装配是被禁用的。
经常使用的自动装配注解有如下几种:@Autowired,@Resource,@Inject,@Qualifier,@Named。@Autowired注解是byType类型的,这个注解能够用在属性上面,setter方面上面以及构造器上面。使用这个注解时,就不须要在类中为属性添加setter方法了。可是这个属性是强制性的,也就是说必须得装配上,若是没有找到合适的bean可以装配上,就会抛出异常。这时可使用required=false来容许能够不被装配上,默认值为true。当required=true时,@Autowired要求必须装配,可是在没有bean能装配上时,就会抛出异常:NoSuchBeanDefinitionException,若是required=false时,则不会抛出异常。另外一种状况是同时有多个bean是一个类型的,也会抛出这个异常。此时须要进一步明确要装配哪个Bean,这时能够组合使用@Qualifier注解,值为Bean的名字便可。@Qualifier注解使用byName进行装配,这样能够在多个类型同样的bean中,明确使用哪个名字的bean来进行装配。@Qualifier注解起到了缩小自动装配候选bean的范围的做用。注意:@Autowired注解是spring提供的,因此会依赖spring的包。还有一个byType的注解@Inject,与@Autowired注解做用同样,也是byType类型,并且是java ee提供的,彻底能够代替@Autowired注解,可是@Inject必须是强制装配的,没有required属性,也就是不能为null,若是不存在匹配的bean,会抛出异常。@Autowired与@Qualifier能够组合使用,@Inject也有一个组合的注解,就是@Named注解,与@Qualifier做用同样,也是byName,可是不是spring的,是java ee标准的。这样就出现了两套自动装配的注解组合,@Autowired与@Qualifier是spring提供的,@Inject与@Named是java ee的。可是@Qualifier注解在java ee中也有同样,做用与spring的@Qualifier注解如出一辙,只是所在的包不同。不过建议你们使用spring的。最后还有一个@Resouce注解, 这个注解也是java ee的,也是byName类型的,原理同@Qualifier和@Named是同样的。
最后说一说,自动检测配置,也是springmvc中最牛的一项功能。只要一个配置<context:component-scan base-package="">,base-package属性指定要自动检测扫描的包。
该配置会自动扫描指定的包及其子包下面被构造型注解标注的类,并将这些类注册为spring bean,这样就不用在配置文件一个一个地配置成bean标签。构造型注解包括:@Controller,@Components,@Service,@Repository和使用@Component标注的自定义注解。生成的bean的ID默认为类的非限定名,也就是把类的名字的首字母换成小写。能够在这些注解的值中写名bean id的值,如@Controller("helloworld")。若是你想细化包被扫描的范围,可使用<context:include-filter>和<context:exclude-filter>。具体使用方法这里再也不详说。注意,没有被扫描到的类是不能注册为bean,也就不能被用来装配其余类。因此这个配置的base-package的范围很是重要。html

 

 本文选自西安楼凤java

相关文章
相关标签/搜索