依赖注入

平常的java开发中,通常是new一个依赖类再调用类实例的方法,这种开发存在的问题是new的类实例不好统一管理,spring提出了依赖注入的思想,即依赖类不由程序员实例化,而是通过spring容器帮我们new指定实例并且将实例注入到需要该对象的类中。依赖注入的另一种说法是“控制反转”,通俗的理解是:平常我们new一个实例,这个实例的控制权是我们程序员,而控制反转是指new实例工作不由我们程序员来做而是交给spring容器来做。

spring有多种依赖注入的形式,下面介绍spring进行DI的方式:

一.目前使用最广泛的 @Autowired:自动装配

自动装配,用于替代基于XML配置的自动装配,详见【3.3.3 自动装配】。

基于@Autowired的自动装配,默认是根据类型注入,可以用于构造器、字段、方法注入,使用方式如下:

java代码:

@Autowired(required=true)
构造器、字段、方法
@Autowired默认是根据参数类型进行自动装配,且必须有一个Bean候选者注入默认required=true,如果允许出现0个Bean候选者需要设置属性“required=false”,“required”属性含义和@Required一样,只是@Required只适用于基于XML配置的setter注入方式,只能打在setting方法上。

(1)、构造器注入:通过将@Autowired注解放在构造器上来完成构造器注入,默认构造器参数通过类型自动装配,如下所示:

1、准备测试Bean,在构造器上添加@AutoWired注解:

java代码:

package cn.javass.spring.chapter12;
import org.springframework.beans.factory.annotation.Autowired;
public class TestBean11 {
private String message;
@Autowired //构造器注入
private TestBean11(String message) {
this.message = message;
}
//省略message的getter和setter
}
2、在Spring配置文件(chapter12/dependecyInjectWithAnnotation.xml)添加如下Bean配置:

xml代码:

3、测试类如下:

java代码:

@Test
public void testAutowiredForConstructor() {
TestBean11 testBean11 = ctx.getBean(“testBean11”, TestBean11.class);
Assert.assertEquals(“hello”, testBean11.getMessage());
}
在Spring配置文件中没有对“testBean11”进行构造器注入和setter注入配置,而是通过在构造器上添加@ Autowired来完成根据参数类型完成构造器注入。

(2)、字段注入:通过将@Autowired注解放在构造器上来完成字段注入。

1、准备测试Bean,在字段上添加@AutoWired注解:

java代码:

package cn.javass.spring.chapter12;
import org.springframework.beans.factory.annotation.Autowired;
public class TestBean12 {
@Autowired //字段注入
private String message;
//省略getter和setter
}
2、在Spring配置文件(chapter12/dependecyInjectWithAnnotation.xml)添加如下Bean配置:

xml代码:

3、测试方法如下:

java代码:

@Test
public void testAutowiredForField() {
TestBean12 testBean12 = ctx.getBean(“testBean12”, TestBean12.class);
Assert.assertEquals(“hello”, testBean12.getMessage());
}
字段注入在基于XML配置中无相应概念,字段注入不支持静态类型字段的注入。

(3)、方法参数注入:通过将@Autowired注解放在方法上来完成方法参数注入。

1、准备测试Bean,在方法上添加@AutoWired注解:

java代码:

package cn.javass.spring.chapter12;
import org.springframework.beans.factory.annotation.Autowired;
public class TestBean13 {
private String message;
@Autowired //setter方法注入
public void setMessage(String message) {
this.message = message;
}
public String getMessage() {
return message;
}
}
java代码:

package cn.javass.spring.chapter12;
//省略import
public class TestBean14 {
private String message;
private List list;
@Autowired(required = true) //任意一个或多个参数方法注入
private void initMessage(String message, ArrayList list) {
this.message = message;
this.list = list;
}
//省略getter和setter
}
2、在Spring配置文件(chapter12/dependecyInjectWithAnnotation.xml)添加如下Bean配置:

xml代码:

代码:

3、测试方法如下:

java代码:

@Test
public void testAutowiredForMethod() {
TestBean13 testBean13 = ctx.getBean(“testBean13”, TestBean13.class);
Assert.assertEquals(“hello”, testBean13.getMessage());

TestBean14 testBean14 = ctx.getBean("testBean14", TestBean14.class);  
Assert.assertEquals("hello", testBean14.getMessage());  
Assert.assertEquals(ctx.getBean("list", List.class), testBean14.getList());  

}
方法参数注入除了支持setter方法注入,还支持1个或多个参数的普通方法注入,在基于XML配置中不支持1个或多个参数的普通方法注入,方法注入不支持静态类型方法的注入。

二.setter 方法注入
这是最简单的注入方式,假设有一个SpringAction,类中需要实例化一个SpringDao对象,那么就可以定义一个private的SpringDao成员变量,然后创建SpringDao的set方法(这是ioc的注入入口):

Java代码

package com.bless.springdemo.action;
public class SpringAction {
//注入对象springDao
private SpringDao springDao;
//一定要写被注入对象的set方法
public void setSpringDao(SpringDao springDao) {
this.springDao = springDao;
}

    public void ok(){  
    springDao.ok();  
}  

}
随后编写spring的xml文件,中的name属性是class属性的一个别名,class属性指类的全名,因为在SpringAction中有一个公共属性Springdao,所以要在标签中创建一个标签指定SpringDao。标签中的name就是SpringAction类中的SpringDao属性名,ref指下面<bean name=“springDao”…>,这样其实是spring将SpringDaoImpl对象实例化并且调用SpringAction的setSpringDao方法将SpringDao注入:

xml代码

<bean name="springAction" class="com.bless.springdemo.action.SpringAction">  
    <!--(1)依赖注入,配置当前类中相应的属性-->  
    <property name="springDao" ref="springDao"></property>  
</bean>  


三:构造器注入

这种方式的注入是指带有参数的构造函数注入,看下面的例子,我创建了两个成员变量SpringDao和User,但是并未设置对象的set方法,所以就不能支持第一种注入方式,这里的注入方式是在SpringAction的构造函数中注入,也就是说在创建SpringAction对象时要将SpringDao和User两个参数值传进来:

Java代码

public class SpringAction {
//注入对象springDao
private SpringDao springDao;
private User user;

public SpringAction(SpringDao springDao,User user){  
    this.springDao = springDao;  
    this.user = user;  
    System.out.println("构造方法调用springDao和user");  
}  
      
    public void save(){  
    user.setName("卡卡");  
    springDao.save(user);  
}  

}
在XML文件中同样不用的形式,而是使用标签,ref属性同样指向其它标签的name属性:

Xml代码

<bean name="springAction" class="com.bless.springdemo.action.SpringAction">  
    <!--(2)创建构造器注入,如果主类有带参的构造方法则需添加此配置-->  
    <constructor-arg ref="springDao"></constructor-arg>  
    <constructor-arg ref="user"></constructor-arg>  
</bean>  
    <bean name="springDao" class="com.bless.springdemo.dao.impl.SpringDaoImpl"></bean>  
     <bean name="user" class="com.bless.springdemo.vo.User"></bean>  

解决构造方法参数的不确定性,你可能会遇到构造方法传入的两参数都是同类型的,为了分清哪个该赋对应值,则需要进行一些小处理:

下面是设置index,就是参数位置:

Xml代码 :

另一种是设置参数类型:

Xml代码 :

四丶静态工厂的方法注入 静态工厂顾名思义,就是通过调用静态工厂的方法来获取自己需要的对象,为了让spring管理所有对象,我们不能直接通过"工程类.静态方法()"来获取对象,而是依然通过spring注入的形式获取:

Java代码

package com.bless.springdemo.factory;

import com.bless.springdemo.dao.FactoryDao;
import com.bless.springdemo.dao.impl.FactoryDaoImpl;
import com.bless.springdemo.dao.impl.StaticFacotryDaoImpl;

public class DaoFactory {
//静态工厂
public static final FactoryDao getStaticFactoryDaoImpl(){
return new StaticFacotryDaoImpl();
}
}
同样看关键类,这里我需要注入一个FactoryDao对象,这里看起来跟第一种注入一模一样,但是看随后的xml会发现有很大差别:

Java代码 :

public class SpringAction {
//注入对象
private FactoryDao staticFactoryDao;

public void staticFactoryOk(){  
    staticFactoryDao.saveFactory();  
}  
//注入对象的set方法  
public void setStaticFactoryDao(FactoryDao staticFactoryDao) {  
    this.staticFactoryDao = staticFactoryDao;  
}  

}
Spring的IOC配置文件,注意看指向的class并不是FactoryDao的实现类,而是指向静态工厂DaoFactory,并且配置 factory-method="getStaticFactoryDaoImpl"指定调用哪个工厂方法:

Xml代码

<bean name="springAction" class="com.bless.springdemo.action.SpringAction" >  
    <!--(3)使用静态工厂的方法注入对象,对应下面的配置文件(3)-->  
    <property name="staticFactoryDao" ref="staticFactoryDao"></property>  
            </property>  
</bean>  
<!--(3)此处获取对象的方式是从工厂类中获取静态方法-->  
<bean name="staticFactoryDao" class="com.bless.springdemo.factory.DaoFactory" factory-method="getStaticFactoryDaoImpl"></bean>  
  五丶实例工厂的方法注入

实例工厂的意思是获取对象实例的方法不是静态的,所以你需要首先new工厂类,再调用普通的实例方法:

Java代码

public class DaoFactory {
//实例工厂
public FactoryDao getFactoryDaoImpl(){
return new FactoryDaoImpl();
}
}
那么下面这个类没什么说的,跟前面也很相似,但是我们需要通过实例工厂类创建FactoryDao对象:

Java代码

public class SpringAction {
//注入对象
private FactoryDao factoryDao;

public void factoryOk(){  
    factoryDao.saveFactory();  
}  

public void setFactoryDao(FactoryDao factoryDao) {  
    this.factoryDao = factoryDao;  
}  

}
最后看spring配置文件:

Xml代码

<bean name="springAction" class="com.bless.springdemo.action.SpringAction">  
    <!--(4)使用实例工厂的方法注入对象,对应下面的配置文件(4)-->  
    <property name="factoryDao" ref="factoryDao"></property>  
</bean>  
  
<!--(4)此处获取对象的方式是从工厂类中获取实例方法-->  
<bean name="daoFactory" class="com.bless.springdemo.factory.DaoFactory"></bean>  
<bean name="factoryDao" factory-bean="daoFactory" factory-method="getFactoryDaoImpl"></bean> 

平常的java开发中,程序员在某个类中需要依赖其它类的方法,则通常是new一个依赖类再调用类实例的方法,这种开发存在的问题是new的类实例不好统一管理,spring提出了依赖注入的思想,即依赖类不由程序员实例化,而是通过spring容器帮我们new指定实例并且将实例注入到需要该对象的类中。依赖注入的另一种说法是“控制反转”,通俗的理解是:平常我们new一个实例,这个实例的控制权是我们程序员,而控制反转是指new实例工作不由我们程序员来做而是交给spring容器来做。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值