1. Spring依赖注入方式
今天这里主要讲解一下Spring框架中的依赖注入的多种方式及相关的一系列配置,这也是Spring这款轻量级DI、IOC的Bean容器框架的核心魅力所在。
Spring依赖注入方式这里着重说明一下构造器参数注入、多类型(Arrays、Set、List、Map、Properties对象)注入、XML自动注入、全注解配置注入这几种注入方式。
下面围绕这几种不同注入方式进行一一讲解:
1.1.构造器参数注入
在Spring(1)知识入门中我们了解到在ApplicationContext.xml文件中配置了bean之后使用SpringTest来获取Bean对象,必须保证Bean所在的类中必须有无参数的构造方法,否则会产生如下错误:
而且配置了无参构造方法之后,获取到的Bean中对应的属性也是null,如下图:
那不需要无参构造方法并且我想注入Bean中的属性值怎么配置呢?
构造器参数注入就解决了上述问题,构造器参数注入的配置方式有三种,如下所示:
① 根据构造器参数索引注入;
<bean id="myBean" class="cn.yif.demo1_constructor.MyBean"> <constructor-arg index="0" value="丽娜">constructor-arg> <constructor-arg index="1" value="21">constructor-arg> bean>
② 根据构造器参数名称注入;
<bean id="myBean" class="cn.yif.demo1_constructor.MyBean"> <constructor-arg name="username" value="丽娜">constructor-arg> <constructor-arg name="age" value="21">constructor-arg> bean>
③ 根据构造器的类型注入;
<bean id="myBean" class="cn.yif.demo1_constructor.MyBean"> <constructor-arg type="java.lang.String" value="丽娜">constructor-arg> <constructor-arg type="java.lang.Integer" value="21">constructor-arg> bean>
④ 特殊情况:如果有一个属性是另外一个类(Bean)的对象,注入有以下两种方式;
方式一:在外部定义,内部构造器使用ref引入;
<bean id="otherBean" class="cn.yif.demo1_constructor.OtherBean"> <property name="otherAge" value="24">property> <property name="otherName" value="李勇">property> bean> <bean id="myBean" class="cn.yif.demo1_constructor.MyBean"> <constructor-arg index="0" value="丽娜">constructor-arg> <constructor-arg index="1" value="21">constructor-arg> <constructor-arg ref="otherBean">constructor-arg> bean>
方式二:在内部constructor-arg标签使用一个完整的bean来注入
<bean id="myBean" class="cn.yif.demo1_constructor.MyBean"> <constructor-arg value="丽娜">constructor-arg> <constructor-arg value="21">constructor-arg> <constructor-arg> <bean id="otherBean" class="cn.yif.demo1_constructor.OtherBean"> <property name="otherAge" value="24">property> <property name="otherName" value="李勇">property> bean> constructor-arg> bean>
1.2.多类型(Arrays、Set、List、Map、Properties对象)注入
① 先准备一个具有多类型属性的MultiTypeBean类,并提供setter与toString方法;
import cn.yif.demo1_constructor.OtherBean; import java.math.BigDecimal; import java.util.*; public class MultiTypeBean { // 简单属性 private Long id; private String name; private Boolean sex; private BigDecimal salary; // 对象属性 private Listlist; private List otherBeanList; private Set set; private Set otherBeanSet; private Map map; //Properties资源属性与String数组属性 private Properties props1; private Properties props2; private String[] arrays; public void setId(Long id) { this.id = id; } public void setName(String name) { this.name = name; } public void setSex(Boolean sex) { this.sex = sex; } public void setSalary(BigDecimal salary) { this.salary = salary; } public void setList(List list) { this.list = list; } public void setOtherBeanList(List otherBeanList) { this.otherBeanList = otherBeanList; } public void setSet(Set set) { this.set = set; } public void setOtherBeanSet(Set otherBeanSet) { this.otherBeanSet = otherBeanSet; } public void setMap(Map map) { this.map = map; } public void setProps1(Properties props1) { this.props1 = props1; } public void setProps2(Properties props2) { this.props2 = props2; } public void setArrays(String[] arrays) { this.arrays = arrays; } @Override public String toString() { return "MultiTypeBean{" + "id=" + id + ", name='" + name + '\'' + ", sex=" + sex + ", salary=" + salary + ", list=" + list + ", otherBeanList=" + otherBeanList + ", set=" + set + ", otherBeanSet=" + otherBeanSet + ", map=" + map + ", props1=" + props1 + ", props2=" + props2 + ", arrays=" + Arrays.toString(arrays) + '}'; } }
1.2.1. Arrays数组注入
写法一,简写方式:
<property name="arrays" value="xxx,yyy,zzz">property>
写法二,正式写法:
<property name="arrays"> <array> <value>xxxvalue> <value>yyyvalue> <value>zzzvalue> array> property>
1.2.2. List集合注入
—对于List>的注入-->
<property name="list">
<list>
<value>aaavalue>
<value>bbbvalue>
<value>cccvalue>
list>
property>
—对于List>的注入-->
<property name="otherBeanList">
<list>
<bean id="otherBean" class="cn.yif.demo1_constructor.OtherBean">bean>
list>
property>
1.2.3. Set集合注入
—对于Set>的注入-->
<property name="set">
<set>
<value>aaavalue>
<value>bbbvalue>
<value>cccvalue>
set>
property>
—对于Set>的注入-->
<property name="otherBeanSet">
<set>
<bean id="otherBean" class="cn.yif.demo1_constructor.OtherBean">bean>
<bean id="otherBean" class="cn.yif.demo1_constructor.OtherBean">bean>
set>
property>
1.2.4. Map字典注入
<property name="map"> <map> <entry key="username" value="李丽">entry> <entry key="age" value="21">entry> map> property>
1.2.5. Properties资源属性注入
<property name="props1"> <value> Jpa.dialect=org.Jpa.dialect.HSQLDialect Jpa.driverClassName=com.mysql.jdbc.Driver value> property> <property name="props2"> <props> <prop key="Jpa.dialect">org.Jpa.dialect.HSQLDialectprop> <prop key="Jpa.driverClassName">com.mysql.jdbc.Driver 支持中文prop> props> property>
1.2.6. 普通字段属性注入
<property name="id" value="1">property> <property name="name" value="李华">property> <property name="sex" value="true">property> <property name="salary" value="8045.5">property>
1.3.XML自动注入
使用XML注入Bean有两种方式注入配置:
① byType方式
这种方式是按照注入对象的类型进行注入,注意:类型要求只能配置一个实例,类型表示接口IUserDao这种对应的实体实现类,配置文件中不能有2个实现类类型,否则会抛出如下异常:
Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [cn.yif.demo3_xmlautoinject.IUserDao] is defined: expected single matching bean but found 2: jdbcDao,jpaDao
② byName方式
这种方式是按照bean的名称,即bean配置中id的值进行注入
③ 配置的使用范围
根节点beans default-autowire="byName" 对当前配置文件的所有bean都生效
子节点bean autowire="byType"只对当前bean生效
具体配置示例如下:
xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd" default-autowire="byType"> <bean id="jdbcDao" class="cn.yif.demo3_xmlautoinject.UserJDBCDaoImpl">bean> <bean id="userService" class="cn.yif.demo3_xmlautoinject.UserServiceImpl" autowire="byName"> bean> <bean id="userAction" class="cn.yif.demo3_xmlautoinject.UserAction"> bean> beans>
1.4.全注解配置
1.4.1.全注解配置使用
在Java中写XML配置Bean还是比较麻烦的,而Spring为了简化还提供了全注解的方式来配置Bean。下面是使用全注解配置Bean的步骤:
① 在ApplicationContext.xml或者当前包路径的SpringXXXTest-Context.xml文件中去配置context命名空间,告诉Spirng去哪个包目录下扫描注解;
xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <context:component-scan base-package="cn.yif.demo4_allannotations" /> <context:annotation-config /> beans>
② Spring提供了四个注解类型:@Component、@Repository、@Service、@Controller用于注解,表示这个类需要被Spring管理为Bean。
一般@Component表示组件,用于不知道是其是在三层结构中的那一层时使用;
@Repository一般用于Dao数据持久层;
@Service一般用于业务层注解;
@Controller一般用于表现层(控制层)。
具体配置如下:
Action层:
@Controller public class UserAction { @Autowired private IUserService userService; public void insert(){ System.out.println("UserAction insert..."); userService.insert(); } }
Service层:
@Service public class UserServiceImpl implements IUserService { @Autowired private IUserDao userJPADaoImpl; public void insert(){ System.out.println("UserService insert..."); userJPADaoImpl.insert(); } }
Dao层:
@Repository public class UserJDBCDaoImpl implements IUserDao { @Override public void insert() { System.out.println("UserJDBCDao insert..."); } }
1.4.2.全注解配置细节
这里全注解配置上有一个隐藏的问题,比如我按真实情况来一个Dao接口,但是这个接口有两个实现IUserDao → UserJDBCDaoImpl/UserJPADaoImpl,在注入的时候就会出现如下错误:
Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [cn.yif.demo4_allannotations.IUserDao] is defined: expected single matching bean but found 2: userJDBCDaoImpl,userJPADaoImpl
这里有两种解决方案:
首先为Dao生成的Bean加上名称,我们调用的时候根据名称调用即可;
@Repository("userJDBCDao") public class UserJDBCDaoImpl implements IUserDao { @Override public void insert() { System.out.println("UserJDBCDao insert..."); } } @Repository("userJPADao") public class UserJPADaoImpl implements IUserDao { @Override public void insert() { System.out.println("UserJPADao insert..."); } }
在调用时,我们可以使用Spring公司提供的注解或者Sun公司提供的注解来调用对应名称下的Bean注解,如下:
① 使用Spring公司提供的一套注解@Qualifier(“配置的Bean名称”)
@Service public class UserServiceImpl implements IUserService { @Autowired @Qualifier("userJPADao") private IUserDao userDao; public void insert(){ System.out.println("UserService insert..."); userDao.insert(); } }
② 使用@Resource(name=”配置的Bean名称”)
@Service public class UserServiceImpl implements IUserService { @Resource(name = "userJDBCDao") private IUserDao userDao; public void insert(){ System.out.println("UserService insert..."); userDao.insert(); } }
总结:现在一般使用全注解配置,比较简单,也比较灵活。