DI(dependency injection)依赖注入模式;依赖注入是指将组件的依赖通过外部以参数或其他形式注入;
再看看
IOC(inversion of control)控制反转模式;控制反转是将组件间的依赖关系从程序内部提到外部来管理;
其实两个说法本质上是一个意思。
不管是依赖注入,还是控制反转,都说明Spring采用动态、灵活的方式来管理各种对象。对象与对象之间的具体实现互相透明。在理解依赖注入之前,看如下这个问题在各种社会形态里如何解决:一个人(Java实例,调用者)需要一双鞋子(Java实例,被调用者)。
(1)原始社会里,几乎没有社会分工。需要鞋子的人(调用者)只能自己去编一双鞋子(被调用者)。对应的情形为:Java程序里的调用者自己来new一个对象
(2)进入工业社会,工厂出现。鞋子不再由普通人完成,而在工厂里被生产出来,此时需要不同鞋子的人(调用者)找到工厂,购买鞋子,无须关心鞋子的制造过程。对应Java程序里简单的工厂设计模式但也需要开发人员在工厂类代码中new出对象,如在方法中根据参数返回接口的各实现类。
(3)进入“按需分配”社会,需要鞋子的人不需要找到工厂,坐在家里发出一个简单指令:需要某种鞋子。鞋子就自然出现在他面前。对应Spring的依赖注入。这里没有开发人员主动去new出对象而是通过在Spring的配置文件中定义相应的调用者和被调用者的bean,然后测试类通过实例化Spring的上下文对象,再通过对象调用getBean(“id名”)来获取相应的调用对象
ApplicationContext ctx = new FileSystemXmlApplicationContext(“applicationcontex.xml”);
依赖注入的三种实现:1.设置注入 2.构造注入 3.注解注入
1.设置注入
设值注入是指通过setter方法传入被调用者的实例。这种注入方式简单、直观,因而在Spring的依赖注入里大量使用。看下面代码,是Person的接口
//定义Person接口
public interface Person
{
//Person接口里定义一个使用鞋子的方法
public void useShoes();
}
然后是Shoes的接口
//定义Shoes接口
public interface Shoes
{
//Shoes接口里有个walk的方法
public void walk();
}
Person的实现类
//Chinese实现Person接口
public class Chinese implements Person
{
//面向Shoes接口编程,而不是具体的实现类
private Shoes shoes;
//默认的构造器
public Chinese()
{}
//设值注入所需的setter方法
public void setShoes(Shoes shoes)
{
this.shoes= shoes;
}
//实现Person接口的useShoes方法
public void useShoes()
{
System.out.println(shoes.walk());
}
}
Shoes的第一个实现类
//Shoes的第一个实现类拖鞋 slipper
public class Slipper implements Shoes
{
//默认构造器
public Slipper()
{}
//实现Shoes接口的walk方法
public String walk()
{
return “拖鞋走路好慢”;
}
}
Shoes的第二个实现类
//Shoes的第二个实现类运动鞋 sneaker
public class Sneaker implements Shoes
{
//默认构造器
public Sneaker()
{}
//实现Shoes接口的walk方法
public String walk()
{
return “运动鞋走路好快”;
}
}
下面采用Spring的配置文件将Person实例和Shoes实例组织在一起。配置文件如下所示:
<!-- 下面是标准的XML文件头 -->
<?xml version=“1.0” encoding=“gb2312”?>
<!-- 下面一行定义Spring的XML配置文件的dtd -->
"http://www.springframework.org/dtd/spring-beans.dtd">
<!-- 以上三行对所有的Spring配置文件都是相同的 -->
<!-- Spring配置文件的根元素 -->
<BEANS>
<!—定义第一bean,该bean的id是chinese, class指定该bean实例的实现类 -->
<BEAN class=lee.Chinese id=chinese>
<!-- property元素用来指定需要容器注入的属性,shoes属性需要容器注入此处是设值注入,因此Chinese类必须拥有setShoes方法 -->
<property name="shoes">
<!-- 此处将另一个bean的引用注入给chinese bean -->
<REF local=“slipper”/>
</property>
</BEAN>
<!-- 定义slipper bean -->
<BEAN class=lee.Slipper id=slipper />
</BEANS>
从配置文件中,可以看到Spring管理bean的灵巧性。bean与bean之间的依赖关系放在配置文件里组织,而不是写在代码里。通过配置文件的 指定,Spring能精确地为每个bean注入属性。因此,配置文件里的bean的class元素,不能仅仅是接口,而必须是真正的实现类。
Spring会自动接管每个bean定义里的property元素定义。Spring会在执行无参数的构造器后、创建默认的bean实例后,调用对应 的setter方法为程序注入属性值。property定义的属性值将不再由该bean来主动创建、管理,而改为被动接收Spring的注入。
每个bean的id属性是该bean的惟一标识,程序通过id属性访问bean,bean与bean的依赖关系也通过id属性完成。
下面看主程序部分:
public class BeanTest
{
//主方法,程序的入口
public static void main(String[] args)throws Exception
{
//因为是独立的应用程序,显式地实例化Spring的上下文。
ApplicationContext ctx = new FileSystemXmlApplicationContext(“applicationcontex.xml”);
//通过Person bean的id来获取bean实例,面向接口编程,因此
//此处强制类型转换为接口类型
Person p = (Person)ctx.getBean(“chinese”);
//直接执行Person的useShoes()方法。
p.useShoes(); //这里方法执行控制台会输出 “拖鞋走路好慢”
}
}
程序的执行结果如下:
拖鞋走路好慢
主程序调用Person的useShoes()方法时,该方法的方法体内需要使用Shoes的实例,但程序里没有任何地方将特定的Person实例和Shoes实 例耦合在一起。或者说,程序里没有为Person实例传入Shoes的实例,Shoes实例由Spring在运行期间动态注入。
Person实例不仅不需要了解Shoes实例的具体实现,甚至无须了解Shoes的创建过程。程序在运行到需要Shoes实例的时候,Spring创建了Shoes实例,然后注入给需要Shoes实例的调用者。Person实例运行到需要Shoes实例的地方,自然就产生了Shoes实例,用来供Person实例使用。
这里如果想把鞋子换成运动鞋(注意这里的运动鞋类以上代码已经写出,如果换其他鞋可以新增类)只需要:
修改原来的Spring配置文件,在其中增加如下一行:
<!-- 定义一个Sneaker bean–>
<BEAN class=lee.Sneaker id=sneaker />
该行重新定义了一个Shoes的实现:Sneaker。然后修改chinese bean的配置,将原来传入slipper的地方改为传入sneaker。也就是将
<REF local=“sneaker”/>
改成
<REF local=“sneaker”/>
此时再次执行程序,将得到如下结果:
运动鞋走路好快
Person与Shoes之间没有任何代码耦合关系,bean与bean之间的依赖关系由Spring管理。采用setter方法为目标bean注入属性的方式,称为***设值注入***。
业务对象的更换变得相当简单,对象与对象之间的依赖关系从代码里分离出来,通过配置文件动态管理。
2.构造注入
所谓构造注入,指通过构造函数来完成依赖关系的设定,而不是通过setter方法。对前面代码Chinese类做简单的修改,修改后的代码如下:
Person的实现类
//Chinese实现Person接口
public class Chinese implements Person
{
//面向Shoes接口编程,而不是具体的实现类
private Shoes shoes;
//默认的构造器
public Chinese()
{}
//构造注入所需的带参数的构造器
public Chinse(Shoes shoes)
{
this.shoes = shoes;
}
//实现Person接口的useShoes方法
public void useShoes()
{
System.out.println(shoes.walk());
}
}
此时无须Chinese类里的setShoes方法,构造Person实例时,Spring为Person实例注入所依赖的Shoes实例。构造注入的配置文件也需做简单的修改,修改后的配置文件如下:
<!-- 下面是标准的XML文件头 -->
<xml version=“1.0” encoding=“gb2312”?>
<!-- 下面一行定义Spring的XML配置文件的dtd -->
"http://www.springframework.org/dtd/spring-beans.dtd">
<!-- 以上三行对所有的Spring配置文件都是相同的 -->
<!-- Spring配置文件的根元素 -->
<BEANS>
<!—定义第一个bean,该bean的id是chinese, class指定该bean实例的实现类 -->
<BEAN class=lee.Chinese id=chinese>
</BEAN>
<!-- 定义sneaker bean -->
<BEAN class=lee.Sneaker id=sneaker />
</BEANS>
执行效果与使用sneaker设值注入时的执行效果完全一样。区别在于:创建Person实例中Axe属性的时机不同——设值注入是现创建一个默认的bean实例,然后调用对应的构造方法注入依赖关系。而构造注入则在创建bean实例时,已经完成了依赖关系
3.注解注入
在进行Spring开发时所需要的基础jar包有:
这里写图片描述
当需要在Spring中使用注解的时候,还需要导入
这里写图片描述
在配置文件中引入新的约束:
xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
在配置文件中进行配置:
在需要使用注解进行的时候,需要在配置文件中进行 开启注解扫描的配置
<context:component-scan base-package=“com.某某包名” ></context:component-scan>
context:annotation-config</context:annotation-config>
使用注解的方式来创建对象:
Computer.java(需要被创建的类)
/**
-
Created by **
*/
@Component(value = “computer”) //相当于在配置文件中
@Scope(value = “prototype”) //配置创建对象是否是以单列模式进行创建
public class Computer {
public void printBrand(){
System.out.println(“小米笔记本”);}
}
在需要实例化的类的类名上面加上@Component 注解来进行标识,value的值就相当于在配置文件中进行配置时bean标签的id属性的值,用于对象的创建。
创建对象的四个注解:
@Component
@Controller //web层中使用
@Service //业务层
@Repository //持久层
这四个注解目前的功能都是一样的,注解名的不同为了能够让标记类本身的用途更加清晰,Spring在后续的版本中会对其加强。
创建对象的测试代码:
@Test
public void Test(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext02.xml");
Computer computer = (Computer) context.getBean("computer");
computer.printBrand();
}
运行结果:
小米笔记本
使用注解注入对象属性
在开发中我们经常会遇到在一个类中持有另外一个类的引用,在前面的博客中记录了使用配置文件中注入对象属性的方式,下面我们接着来看一下使用注解进行注入的方式:
下面的例子,在UserService中持有UserDao这个对象的引用
UserDao.java
/**
-
Created by **
*/
@Component(value = “userDao”)
public class UserDao {public void printInfo(){
System.out.println("UserDao中的实例化对象里的方法执行了");
}
}
两种对象属性注入的注解:
1.使用 @Autowired 注解进行自动装配,不需要指定要注入的对象的value值,自动的根据类名去寻找对应的类来创建对象并进行对象属性的注入。
2. 使用 @Resource(name=”userDao”),需要指定需要创建的对象的名字,这里的name对应@Component注解中的value的值,使用这个注解能够根据我们所指定的对象名准确创建出我们所需要的对象。
UserService.java(示例中使用@Resource注解进行对象属性的注入)
/**
-
Created by **
*/
@Service(value = “userService”)
public class UserService {private List list;
//@Autowired //自动装配,根据类名称去找相应的类来创建对象
@Resource(name=“userDao”) //name属性值写使用注解创建对象时的value中的值
private UserDao userDao;public void print(){
userDao.printInfo();
}
}
MyTest.java 测试代码:
@Test
public void Test(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService service = (UserService) context.getBean("userService");
service.print();
}
运行结果:
UserDao中的实例化对象里的方法执行了
配置文件和注解混合使用
1.创建对象的操作使用配置文件的方式实现
2.注入属性的操作使用注解的方式实现
下面看一下混合使用的例子:
在StudentService中持有BookDao,ComputerDao对象的引用,调用StudentService实例化对象中的printDo()方法测试对象是否创建成功。
BookDao.java
/**
-
Created by **
*/
public class BookDao {public void doSomething(){
System.out.println("读书");
}
}
ComputerDao.java
/**
-
Created by **
*/
public class ComputerDao {public void doSomething(){
System.out.println("敲代码……");
}
}
StudentService.java
/**
-
Created by **
*/
public class StudentService {
@Resource(name = “bookDao”)
private BookDao bookDao;
@Resource(name = “computerDao”)
private ComputerDao computerDao;public void printDo(){
bookDao.doSomething();
computerDao.doSomething();
}
}
applicationContext.xml(Spring 配置文件)
<context:component-scan base-package=“com.某某包”></context:component-scan>
MyTest.java(测试代码)
@Test
public void Test(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
StudentService service = (StudentService) context.getBean("studentService");
service.printDo();
}
运行结果:
读书
敲代码……