Type1 接口注入
我们常常借助接口来将调用者与实现者分离。如:
public class ClassA {
private InterfaceB clzB;
public init() {
Ojbect obj =
Class.forName(Config.BImplementation).newInstance();
clzB = (InterfaceB)obj;
}
……
}
上面的代码中,ClassA依赖于InterfaceB的实现,如何获得InterfaceB实现类的实例?传统的方法是在代码中创建InterfaceB实现类的实例,并将起赋予clzB.
而这样一来,ClassA在编译期即依赖于InterfaceB的实现。为了将调用者与实现者在编译期分离,于是有了上面的代码,我们根据预先在配置文件中设定的实现类的类名,动态加载实现类,并通过InterfaceB强制转型后为ClassA所用。
这就是接口注入的一个最原始的雏形。
而对于一个Type1型IOC容器而言,加载接口实现并创建其实例的工作由容器完成,如J2EE开发中常用的Context.lookup(ServletContext.getXXX),都是Type1型IOC的表现形式。
Apache Avalon是一个典型的Type1型IOC容器。
Type2 构造子注入
构造子注入,即通过构造函数完成依赖关系的设定,如:
public class DIByConstructor {
private final DataSource dataSource;
private final String message;
public DIByConstructor(DataSource ds, String msg) {
this.dataSource = ds;
this.message = msg;
}
……
}
可以看到,在Type2类型的依赖注入机制中,依赖关系是通过类构造函数建立,容器通过调用类的构造方法,将其所需的依赖关系注入其中。
PicoContainer(另一种实现了依赖注入模式的轻量级容器)首先实现了Type2类型的依赖注入模式。
Type3 设值注入(用setter,getter设值)
在各种类型的依赖注入模式中,设值注入模式在实际开发中得到了最广泛的应用(其中很大一部分得力于Spring框架的影响)。
在笔者看来,基于设置模式的依赖注入机制更加直观、也更加自然。Quick Start中的示例,就是典型的设置注入,即通过类的setter方法完成依赖关系的设置。
SpringFrameWork Developer’s Guide Version 0.6
September 2, 2004 So many open source projects. Why not Open your Documents?
几种依赖注入模式的对比总结
接口注入模式因为具备侵入性,它要求组件必须与特定的接口相关联,因此并不被看好,实际使用有限。
Type2和Type3的依赖注入实现模式均具备无侵入性的特点。在笔者看来,这两种实现方式各有特点,也各具优势(一句经典废话?)。
Type2 构造子注入的优势:
1. “在构造期即创建一个完整、合法的对象”,对于这条Java设计原则,Type2无疑是最好的响应者。
我的理解:就是你要通过一种方式来保证对象的引用完整性,type2选择了构造器的方式来实现。
2. 避免了繁琐的setter方法的编写,所有依赖关系均在构造函数中设定,依赖关系集中呈现,更加易读。
我的理解:使用构造方法就不需要每个属性都写set和get方法了,这样省去了很多的代码。
3. 由于没有setter方法,依赖关系在构造时由容器一次性设定,因此组件在被创建之后即处于相对“不变”的稳定状态,无需担心上层代码在调用过程中执行setter方法对组件依赖关系产生破坏,特别是对于Singleton模式的组件而言,这可能对整个系统产生重大的影响。
我的理解:使用构造器来实现,那么你需要一次对所有的属性都初始话,相对set方法来说,缺少了一些灵活性。
4. 同样,由于关联关系仅在构造函数中表达,只有组件创建者需要关心组件内部的依赖关系。对调用者而言,组件中的依赖关系处于黑盒之中。对上层屏蔽不必要的信息,也为系统的层次清晰性提供了保证。
我的理解: spring的这设计就是要屏蔽依赖关系,你只需要对接口编程,而不需要考虑依赖关系的实现。所以对调用者来说,依赖关系是处于黑盒当中。
5. 通过构造子注入,意味着我们可以在构造函数中决定依赖关系的注入顺序,对于一个大量依赖外部服务的组件而言,依赖关系的获得顺序可能非常重要,比如某个依赖关系注入的先决条件是组件的DataSource及相关资源已经被设定。
我的理解:关于顺序问题,我们来看以下两段代码:
public DIByConstructor(DataSource ds, String msg) {
this.dataSource = ds;
this.message = msg;
}
public DIByConstructor(DataSource ds, String msg) {
this.dataSource = ds;
this.message = msg;
}
在本例中,顺序不太重要,但是如果message的初始化需要用到datasource 的话,那么就必须要先初始化datasource,所以相对来说,顺序就是确定了。
Type3 设值注入的优势
1. 对于习惯了传统JavaBean开发的程序员而言,通过setter方法设定依赖关系显得更加直观,更加自然。
2. 如果依赖关系(或继承关系)较为复杂,那么Type2模式的构造函数也会相当庞大(我们需要在构造函数中设定所有依赖关系),此时Type3模式往往更为简洁。
我的理解:依赖关系(或继承关系)较为复杂,指的是,属性较多,需要写很多的set和get方法。
3. 对于某些第三方类库而言,可能要求我们的组件必须提供一个默认的构造函数(如Struts中的Action),此时Type2类型的依赖注入机制就体现出其局限性,难以完成我们期望的功能。
可见,Type2和Type3模式各有千秋,而Spring、PicoContainer都对Type2和Type3类型的依赖注入机制提供了良好支持。这也就为我们提供了更多的选择余地。理论上,以Type2类型为主,辅之以Type3类型机制作为补充,可以达到最好的依赖注入效果,不过对于基于Spring Framework开发的应用而言,Type3使用更加广泛。 |
相关推荐
下面我们将对 Spring Boot 中的几种注入方法进行详细的介绍和分析。 1. @Autowired @Autowired 是 Spring 中最常用的注入方法之一,它可以对类成员变量、方法及构造函数进行标注,完成自动装配的工作。@Autowired ...
在Spring中,DI主要通过两种方式实现:构造器注入和setter注入。构造器注入是在创建对象时通过构造函数传入依赖,而setter注入则是在对象创建后通过setter方法设置依赖。 当我们使用`@Autowired`注解时,Spring容器...
3. **依赖注入**:深入解析依赖注入的概念,如何通过构造器注入、设值注入和接口注入等方式实现对象间的依赖关系。 4. **AOP**:介绍AOP的基本概念,包括切面、通知、连接点、切入点表达式等,并演示如何使用Spring...
5. **Spring MVC**:Spring的Web MVC框架提供了一种灵活的方式来构建RESTful Web服务。源码中包含了DispatcherServlet、Controller、ModelAndView等关键组件的工作原理。 6. **数据访问**:Spring JDBC和ORM模块...
首先,`@Autowired`是Spring框架提供的一种依赖注入(DI,Dependency Injection)方式,用于自动装配bean的依赖。当我们在类的属性或方法上使用这个注解时,Spring会自动寻找匹配类型的bean并将其注入。这种方式比传统...
另一种方式是直接使用`HibernateTemplate`实例。在这种情况下,可以直接注入`HibernateTemplate`并调用其方法进行数据库操作。 **示例代码**: ```java public class UserDaoHibernateImpl implements UserDao { @...
Spring 框架是Java开发中的一个核心框架,尤其对于初学者来说,它是一个极好的起点,可以帮助理解企业级应用的构建方式。本项目旨在为初学者提供一个基础的Spring项目实例,帮助大家快速上手并熟悉Spring的核心概念...
Spring JDBC是Spring框架的一部分,它提供了一种抽象层,使得开发者可以更方便地使用Java数据库连接(JDBC)来操作数据库。在这个实例源码中,我们看到的是一个基于`epet`宠物项目的完整实现,该项目可能包括了创建...
下面将对 Spring 源码中 Bean 对象的创建过程进行分析和解释。 1. 创建工厂,到达 BeanDefinition 步骤 在 Spring 框架中,Bean 对象的创建过程从创建工厂开始。工厂是 Spring 框架的核心组件,负责创建和管理 ...
《Spring技术内幕:深入解析Spring架构与设计原理(第2版)》这本书主要聚焦于Spring框架的核心架构和技术细节,帮助读者全面理解Spring的工作机制、设计理念以及实现方式。下面将根据书名及其描述来展开相关知识点。 ...
本文主要基于Spring 5源码分析(第2版),详细介绍在Spring框架中常见的几种设计模式及其应用场景。 #### 二、设计模式概述 设计模式是一种在特定上下文中解决常见设计问题的方法论。在软件工程中,设计模式不仅...
通过对Spring框架设计理念与设计模式的分析,我们可以看到Spring框架的成功在于其对软件工程原则的深刻理解和灵活应用。Spring不仅仅是一个简单的框架,它还是一个设计理念的典范,为开发者提供了强大而灵活的工具,...
4. **模板方法模式**:Spring框架中的某些组件采用了模板方法模式,比如JdbcTemplate等,为开发者提供了一种通用的操作数据库的方式。 5. **观察者模式**:Spring事件传播机制使用了观察者模式,使得组件之间能够...
获取Bean主要有以下几种方式: 1. **通过名称获取Bean** 使用`ApplicationContext`的`getBean(String name)`方法可以直接根据Bean的定义名称获取到对应的实例。例如: ```java ApplicationContext context = new...
在Spring 1.2.6源码中,我们可以看到以下几个关键知识点: 1. **依赖注入(Dependency Injection,DI)**:这是Spring的核心功能之一,它通过反转控制来管理对象之间的依赖关系。源码中,你可以看到`BeanFactory`...
在 JCatalog 示例中,Spring 可能用于实现业务服务接口,提供对 Hibernate DAO 的封装,而 JSF 的 Managed Beans 作为视图与控制器的结合,负责处理用户请求并调用 Spring 服务。此外,安全控制也应纳入设计考虑,如...
在"OKEC.Sample.Spring"这个项目中,通过逐步分析源代码、配置文件以及运行结果,你可以亲身体验这些概念是如何在实际项目中工作的,从而更好地理解和应用Spring.NET框架。同时,这也是一个实践动手的好机会,可以...
1. **通过全类名**:这是最基础也是最原始的配置方式,开发者需要在配置文件中指定类的全限定名称,Spring会通过反射机制创建Bean的实例。 2. **通过工厂方法**:包括静态工厂方法和实例工厂方法。这种方式允许...
Spring提供了多种DI方式,包括构造器注入、setter注入和接口注入。开发者可以通过注解或XML配置来指定依赖关系。 **3. 模拟Spring的IOC容器** 要模拟Spring的IOC容器,我们需要实现以下几个核心功能: - **Bean...