【java框架】Spring(2) -- Spring依赖注入配置

2020-03-10 16:04:26来源:博客园 阅读 ()

新老客户大回馈,云服务器低至5折

【java框架】Spring(2) -- Spring依赖注入配置

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">
<!--方式一:根据构造器参数的索引注入 index:从0开始,属性位置; value:设置值-->
<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">
<!--方式二:根据构造器的参数名称注入 name:属性名; value:设置值-->
<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">
<!--方式三:根据构造器的类型注入  type:类型全限定名; value:设置值-->
<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>
<!--方式一:otherBean在外部定义好,内部constructor-arg使用ref引入-->
<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来注入

<!--方式二:在constructor-arg内部使用bean标签定义一个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 List<String> list;
private List<OtherBean> otherBeanList;
private Set<String> set;
private Set<OtherBean> otherBeanSet;
private Map<String,Object> 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<String> list) {
    this.list = list;
}

public void setOtherBeanList(List<OtherBean> otherBeanList) {
    this.otherBeanList = otherBeanList;
}

public void setSet(Set<String> set) {
    this.set = set;
}

public void setOtherBeanSet(Set<OtherBean> otherBeanSet) {
    this.otherBeanSet = otherBeanSet;
}

public void setMap(Map<String, Object> 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>xxx</value>
      <value>yyy</value>
      <value>zzz</value>
  </array>
</property>

1.2.2.     List集合注入

<!—对于List<String>的注入-->
<property name="list">
 <list>
     <value>aaa</value>
     <value>bbb</value>
     <value>ccc</value>
  </list>
</property>
<!—对于List<OtherBean>的注入-->
<property name="otherBeanList">
 <list>
         <bean id="otherBean" class="cn.yif.demo1_constructor.OtherBean"></bean>
 </list>
</property>

1.2.3.     Set集合注入

<!—对于Set<String>的注入-->
<property name="set">
  <set>
      <value>aaa</value>
      <value>bbb</value>
      <value>ccc</value>
  </set>
</property>
<!—对于Set<OtherBean>的注入-->
<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.HSQLDialect</prop>
      <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">
    <!--default-autowire="byType":表示是全局的注入方式,所有bean通过byType方式注入-->
    <bean id="jdbcDao" class="cn.yif.demo3_xmlautoinject.UserJDBCDaoImpl"></bean>
    <!--<bean id="jpaDao" class="cn.yif.demo3_xmlautoinject.UserJPADaoImpl" autowire="byName"></bean>-->
    <!--userService需要userDao这个类型,我通过byType这个类型注入给你一个就行-->
    <!--autowire="byName"表示这个bean的注入方式由我自定义注入,我通过byName方式注入-->
    <bean id="userService" class="cn.yif.demo3_xmlautoinject.UserServiceImpl" autowire="byName">
    </bean>
    <!--userAction需要userService这个实现类的属性,我通过byType这个类型给你注入就行-->
   <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" />
<!-- 这个不是必须的(spring3.2版本前使用) 配上后兼容性好 -->
<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();
}
}

总结:现在一般使用全注解配置,比较简单,也比较灵活。


原文链接:https://www.cnblogs.com/yif0118/p/12459494.html
如有疑问请与原作者联系

标签:

版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有

上一篇:项目系统Netty的Channel和用户之间的关系绑定正确做法,以及Chann

下一篇:java工作流引擎:jbpm和activiti对比分析