`

spring 控制反转与依赖注入原理-学习笔记

阅读更多

 

在Spring中有两个非常重要的概念,控制反转和依赖注入;控制反转将依赖对象的创建和管理交由Spring容器,而依赖注入则是在控制反转的基础上将Spring容器管理的依赖对象注入到应用之中;

所谓依赖注入:在运行期,由外部容器动态将依赖对象注入到组件中。

XML文件解析 + Java反射技术;

首先是XML文件的解析(dom4j),Spring框架对于配置文件的选择是XML文件,根据Spring的规范,配置文件的命名是没有特殊要求的,只是在文件的放置位置上有两种选择;类路径下或者操作系统文件目录下(大多数情况是放到类路径下)。

对于Spring的控制反转和依赖注入来说,唯一使用的是配置文件中的<bean>标签,通过这个标签,Spring就完成了对象的创建和依赖对象的注入工作;

1、首先对于配置文件中的<bean>节点,在Spring框架中存在一个对用的定义接口,叫做BeanDefinition;子啊个类定义了获得<bean>节点中出现的所有属性的方法,例如classNam、scope、factory-method、lazy-init 等等属性;

2、对于<bean>节点的子节点property则完成了属性注入的功能;属性注入有三种方式,构造器注入、属性setter方法注入和注解方式注入;

3、如果是setter方法注入,对于类属性XML配置文件中有两种方法,一是使用property节点的ref属性,一是使用property几点的子节点bean进行内部bean配置;如果是对于基本数据类型进行配置,那么要是用property节点的value属性;


定义自己的关于bean节点、property节点的pojo类文件;

使用注入DOM4J等开源包讲配置文件解析读入;

使用Java的反射技术讲配置文件中的信息setter到我们需要的属性中去;common-beanutils.jar

 

 

	<context:component-scan base-package="com.sample"/>
	<bean id="personService" class="com.spring.junit.test.impl.PersonServiceImpl"></bean>
	<bean id="stockService" class="com.spring.junit.test.impl.StockServiceImpl"></bean>
	
	<bean id="personServiceFactory" class="com.spring.junit.test.impl.PersonServiceBeanFactory" factory-method="createPersonServiceBeanFactory"></bean>
	
	<bean id="personServiceFactory2" class="com.spring.junit.test.impl.PersonServiceBeanFactory"></bean>
	<bean id="stockServiceFactory" factory-bean="personServiceFactory2" factory-method="createStockServiceBeanFactory"></bean>
	
	<bean id="randomBean" class="com.spring.junit.bean.StaticFactoryBean" factory-method="createRandom" scope="prototype"></bean>

	<!-- 集合类型的注入 -->

	通过setter方法注入
	<bean id="user" class="com.sample.bean.User"></bean>
	<bean id="personDao" class="com.sample.dao.impl.PersonDaoBeanImpl"></bean>
	<bean id="personService" class="com.sample.service.impl.PersonServiceBeanImpl">
	   
	   <property name="personDao" ref="personDao"></property>
	   <property name="name" value="jackjson_xu_test"></property>
	   <property name="id" value="108"></property>
	   <property name="sets">
	       <set>
	           <value>第一个</value>
	           <value>第二个</value>
	           <value>第三个</value>
	       </set>
	   </property>
	   <property name="lists">
	       <list>
	           <value>第一個list元素</value>
	           <value>第二個list元素</value>
	           <value>第三個list元素</value>
	           
	       </list>
	   </property>
	   <property name="properties">
	       <props>
	           <prop key="key1">value1</prop>
	           <prop key="key2">value2</prop>
	           <prop key="key3">value3</prop>
	           
	       </props>
	   </property>
	   <property name="maps">
	       <map>
	           <entry key="key-1" value="value-1"></entry>
	           <entry key="key-2" value="value-2"></entry>
	           <entry key="key-3" value="value-3"></entry>
	           <entry key="key-4" value="value-4"></entry>
	       </map>
	   </property>
	   <property name="users">
	       <map>
	           <entry key="U_1001">
	               <ref bean="user"/>
	           </entry>
	           <entry key="U_1002">
                   <ref bean="user"/>
               </entry>
	       </map>
	   </property>
	</bean>
	<!-- 采用内部bean的方式注入 -->

	<bean id="personService" class="com.sample.service.impl.PersonServiceBeanImpl">
	   <property name="personDao">
	       <bean class="com.sample.dao.impl.PersonDaoBeanImpl"/>
	   </property>
	   <property name="name" value="jackjson_xu_test"></property>
	   <property name="id" value="100"></property>
	</bean>
	
	
	<!-- 构造器注入方式 -->
	
	<bean id="personDao" class="com.sample.dao.impl.PersonDaoBeanImpl"></bean>
    <bean id="personService2" class="com.sample.service.impl.PersonServiceBeanImpl2" autowire="byType">
        <constructor-arg index="0" type="com.sample.dao.IPersonDao" ref="personDao"></constructor-arg>
        <constructor-arg index="1" type="java.lang.String" value="http://www.woyo.com"></constructor-arg>
    </bean>
  

 package com.sample.junit;

import java.util.ArrayList;
import java.util.List;
/**
 * Spring xml 属性的方法
 * @author DY
 *
 */
public class BeanDefinition {
	private String id;
	private String className;
	private List<PropertyDefinition> propertys = new ArrayList<PropertyDefinition>();

	public BeanDefinition(String id, String className) {
		this.id = id;
		this.className = className;
	}

	public String getId() {
		return id;
	}

	public void setId(String id) {
		this.id = id;
	}

	public String getClassName() {
		return className;
	}

	public void setClassName(String className) {
		this.className = className;
	}

	public List<PropertyDefinition> getPropertys() {
		return propertys;
	}

	public void setPropertys(List<PropertyDefinition> propertys) {
		this.propertys = propertys;
	}
}

 package com.sample.junit;

/**
 * Spring xml bean 子节点property属性方法
 * 
 * @author DY
 * 
 */
public class PropertyDefinition {
	private String name;
	private String ref;
	private String value;

	public PropertyDefinition(String name, String ref, String value) {
		this.name = name;
		this.ref = ref;
		this.value = value;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public String getRef() {
		return ref;
	}

	public void setRef(String ref) {
		this.ref = ref;
	}

	public String getValue() {
		return value;
	}

	public void setValue(String value) {
		this.value = value;
	}

}

 package com.sample.junit;

import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.apache.commons.beanutils.ConvertUtils;
import org.apache.log4j.Logger;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.XPath;
import org.dom4j.io.SAXReader;
/**
 * spring装配applicationContext.xml文件
 * @author DY
 *
 */
public class SampleClassPathXMLApplicationContext {
	private Logger logger = Logger.getLogger(SampleClassPathXMLApplicationContext.class);
	private List<BeanDefinition> beanDefines = new ArrayList<BeanDefinition>();
	private Map<String, Object> sigletons = new HashMap<String, Object>();

	public SampleClassPathXMLApplicationContext(String filename) {
		this.readXML(filename);
		this.instanceBeans();	//bean的实例化 Class.forName().newInstance()
		this.annotationInject();//注解
		this.injectObject();	//bean对象的属性注入值
	}
	/**
	 * 注解处理器
	 * 如果注解SampleResouce配置了name属性,则根据name所指定的名称获取要注入的实例引用
	 * 如果注解SampleResouce没有配置name属性,则根据属性所属类型来扫描配置文件获取要注入的实例引用
	 */
	private void annotationInject() {
		for (String beanName : sigletons.keySet()) {
			Object bean = sigletons.get(beanName);
			if (bean != null) {
				this.propertyAnnotation(bean);
				this.fieldAnnotation(bean);
			}
		}
	}
	/**
	 * 处理在所有set方法加入的注解
	 * @param bean 处理的bean对象
	 */
	private void propertyAnnotation(Object bean) {
		try {
			//获取其属性的描述
			PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();
			for (PropertyDescriptor properdesc : ps) {
				//获取属性的setter方法
				Method setter = properdesc.getWriteMethod();
				//setter方法上是否存在注解
				if (setter != null && setter.isAnnotationPresent(SampleResource.class)) {
					//获取当前注解,判断name属性是否为空
					SampleResource resouce = setter.getAnnotation(SampleResource.class);
					Object value = null;
					if (resouce.name() != null && !"".equals(resouce.name())) {
						value = sigletons.get(resouce.name());
						setter.setAccessible(true);
						setter.invoke(bean, value);//把引用对象注入到属性
					} else {//如果当前属性没有指定name,则根据类型匹配
						value = sigletons.get(resouce.name());
						if (value == null) {
							for (String key : sigletons.keySet()) {
								//判断当前属性所属类型是否在配置文件中存在
								if (properdesc.getPropertyType().isAssignableFrom(sigletons.get(key).getClass())) {
									value = sigletons.get(key);	//获取类型匹配的实例对象
								}
							}
						}
						//允许访问private方法
						setter.setAccessible(true);
						//把引用对象注入属性
						setter.invoke(bean, value);
					}
				}
			}
		} catch (Exception e) {
			logger.error(e.getLocalizedMessage());
		}
	}
	/**
	 * 处理在字段上的注解
	 * @param bean
	 */
	private void fieldAnnotation (Object bean) {
		try {
			//获取全部属性对象数组
			Field[] fields = bean.getClass().getFields();
			for (Field field : fields) {
				if (field.isAnnotationPresent(SampleResource.class)) {
					SampleResource resouce = field.getAnnotation(SampleResource.class);
					Object value = null;
					if (resouce.name() != null && !"".equals(resouce.name())) {
						value = sigletons.get(resouce.name());
					} else {
						value = sigletons.get(field.getName());
						if (value == null) {
							for (String key : sigletons.keySet()) {
								//根据字段类型匹配
								if (field.getType().isAssignableFrom(sigletons.get(key).getClass())) {
									value = sigletons.get(key);
									break;
								}
							}
						}
					}
					field.setAccessible(true);
					field.set(bean, value);
				}
			}
		} catch (Exception e) {
			e.getLocalizedMessage();
			logger.error("字段注解解析异常:" + e.getLocalizedMessage());
		}
	}
	/**
	 * 为bean对象的属性注入值
	 */
	private void injectObject() {
		for (BeanDefinition beanDefinition : beanDefines) {
			Object bean = sigletons.get(beanDefinition.getId());
			if (bean != null) {
				try {
					PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();
					for (PropertyDefinition propertyDefinition : beanDefinition.getPropertys()) {
						for (PropertyDescriptor properdesc : ps) {
							if (propertyDefinition.getName().equals(properdesc.getName())) {
								Method setter = properdesc.getWriteMethod();// 获取属性的setter方法
								if (setter != null) {
									Object value = null;
									if (propertyDefinition.getRef() != null && !"".equals(propertyDefinition.getRef().trim())) {
										value = sigletons.get(propertyDefinition.getRef());
									} else {
										value = ConvertUtils.convert(propertyDefinition.getValue(), properdesc.getPropertyType());
									}
									setter.setAccessible(true);//私有方法给与访问权限
									setter.invoke(bean, value);// 把引用对象注入到属性
								}
								break;
							}
						}
					}
				} catch (Exception e) {
				}
			}
		}
	}

	/**
	 * 完成bean的实例化
	 */
	private void instanceBeans() {
		for (BeanDefinition beanDefinition : beanDefines) {
			try {
				if (beanDefinition.getClassName() != null
						&& !"".equals(beanDefinition.getClassName().trim()))
					sigletons.put(beanDefinition.getId(), Class.forName(beanDefinition.getClassName()).newInstance());
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
	}

	/**
	 * 读取xml配置文件
	 * 
	 * @param filename
	 */
	private void readXML(String filename) {
		SAXReader saxReader = new SAXReader();
		Document document = null;
		try {
			URL xmlpath = this.getClass().getClassLoader().getResource(filename);
			document = saxReader.read(xmlpath);
			Map<String, String> nsMap = new HashMap<String, String>();
			nsMap.put("ns", "http://www.springframework.org/schema/beans");// 加入命名空间
			XPath xsub = document.createXPath("//ns:beans/ns:bean");// 创建beans/bean查询路径
			xsub.setNamespaceURIs(nsMap);// 设置命名空间
			List<Element> beans = xsub.selectNodes(document);// 获取文档下所有bean节点
			for (Element element : beans) {
				String id = element.attributeValue("id");// 获取id属性值
				String clazz = element.attributeValue("class"); // 获取class属性值
				BeanDefinition beanDefine = new BeanDefinition(id, clazz);
				XPath propertysub = element.createXPath("ns:property");
				propertysub.setNamespaceURIs(nsMap);// 设置命名空间
				List<Element> propertys = propertysub.selectNodes(element);
				for (Element property : propertys) {
					String propertyName = property.attributeValue("name");
					String propertyref = property.attributeValue("ref");
					String propertyValue = property.attributeValue("value");
					PropertyDefinition propertyDefinition = new PropertyDefinition(propertyName, propertyref, propertyValue);
					beanDefine.getPropertys().add(propertyDefinition);
					System.out.println("propertyName:" + propertyName + "|propertyref:" + propertyref + "|propertyValue:" + propertyValue);
				}
				beanDefines.add(beanDefine);
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

	/**
	 * 获取bean实例
	 * 
	 * @param beanName
	 * @return
	 */
	public Object getBean(String beanName) {
		return this.sigletons.get(beanName);
	}
}

 package com.sample.junit;

import org.junit.BeforeClass;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.sample.service.IPersonService;

public class SpringTest {

	static ApplicationContext ctx = null;
	@BeforeClass
	public static void setUpBeforeClass() throws Exception {
		ctx = new ClassPathXmlApplicationContext(new String[]{"applicationContext.xml"});
	}
	
	
	@Test public void instanceSpring(){
		IPersonService personService = (IPersonService)ctx.getBean("personService");
		System.out.println(personService);
		personService.save();
	}
	

}
分享到:
评论
2 楼 u011028234 2016-12-02  
楼主你这例子里边的SampleResource实体没有额?
1 楼 yueerba 2016-11-16  
[flash=200,200][url][url][img][list]
[*]
引用
[u][i][b][/b][/i][/u]
[/list][/img][/url][/url][/flash]
|

相关推荐

    Spring.NET学习笔记-控制反转

    ### Spring.NET 学习笔记 — 控制反转 #### 一、控制反转(IoC)概念解析 控制反转(Inversion of Control,简称IoC)是一种软件设计思想,它改变了传统对象之间的依赖关系管理方式,从而降低了组件之间的耦合度。在...

    Spring - 学习笔记

    Spring 框架是Java开发中的一个重要组成部分,它是一个开源的控制反转(IoC)和面向切面(AOP)的容器框架。这个框架的主要目标是简化企业级应用程序的开发过程,通过提供一种更加灵活和可扩展的方式来管理对象的生命...

    spring ioc aop mvc boot-学习笔记.docx

    Spring框架是Java开发中不可或缺的一部分,它为开发者提供了强大的依赖注入(IOC)和面向切面编程(AOP)功能,以及用于构建Web应用程序的MVC框架。Spring Boot则是基于Spring框架构建的应用程序启动器,旨在简化...

    我的Pro Spring 学习笔记 之二 控制反转(IoC)和依赖注入(DI), Spring初步

    在本文中,我们将深入探讨Spring框架的核心概念——控制反转(Inversion of Control,IoC)和依赖注入(Dependency Injection,DI)。这些概念是Spring框架的基础,对于理解和掌握Spring的使用至关重要。 首先,让...

    spring指南学习笔记

    标题和描述均提到了“spring指南学习笔记”,这意味着文档聚焦于Spring框架的学习心得与关键概念。Spring是一个开源的Java企业级应用框架,以其强大的依赖注入(Dependency Injection, DI)和面向切面编程(Aspect ...

    Spring.NET学习笔记 - 刘冬.NET - 博客园

    Spring.NET是中国.NET开发者社区中广泛讨论的一个开源框架,它为.NET平台带来了与Java Spring相似的功能,包括依赖注入(DI)和面向切面编程(AOP)。这个压缩包包含了一系列刘冬.NET在博客园上发布的Spring.NET学习...

    Spring学习笔记(精华全记录)

    ### Spring学习笔记(精华全记录) #### Spring框架概述 Spring框架源自Rod Johnson的个人项目,最初于2002年末发布。Spring并非一开始就作为一个完整的框架出现,而是从一个项目逐步发展而来。随着项目的成熟,...

    Spring框架学习笔记

    Spring的核心是IOC(Inversion of Control)容器,通过控制反转实现对象之间的解耦,使得代码更加灵活和可测试。 二、依赖注入(DI) 依赖注入是Spring的核心功能之一,它允许开发者在运行时动态地将依赖关系注入到...

    Spring&Mybatis&SpringMVC总结笔记-最全最基础.pdf

    IoC(控制反转)和DI(依赖注入)是Spring的核心概念,它们通过反转控制(将对象的创建和管理交给容器进行)来实现依赖关系的自动注入。AOP(面向切面编程)允许开发者定义方法执行前后及抛出异常后的特定逻辑,它...

    spring特点和原理.学习笔记

    本文将深入探讨Spring的核心特点——控制反转(IoC)和面向方面编程(AOP),以及它们在Spring框架中的实现原理。 **一、控制反转(IoC)** 控制反转(Inversion of Control,简称IoC)是Spring的核心概念之一,它...

    Spring的学习笔记

    Spring通过IOC(控制反转)和DI(依赖注入)实现了对对象生命周期的管理,从而消除了手动管理的复杂性。 **搭建Spring运行环境** 1. **建立新项目**:创建一个新的Java项目,通常会使用Maven或Gradle作为构建工具...

    spring学习笔记1

    ### Spring学习笔记1 #### 一、Spring框架简介与IOC概念 **Spring**是一个开源的Java平台框架,它提供了一种全面的编程模型,并通过一系列的功能模块支持企业级应用程序开发。Spring的核心特性包括依赖注入...

    Spring学习笔记.doc

    ### Spring学习笔记知识点详解 #### 一、Spring框架概述 **1.1 什么是Spring** Spring框架是一个开源的轻量级应用框架,主要用于简化企业级应用程序的开发过程。它的核心特性在于提供了一种灵活的方式来组织和...

    学习Spring笔记_IoC(控制反转)简介

    **Spring框架中的IoC(控制反转)概念** 在软件开发中,IoC(Inversion of Control,控制反转)是一种设计模式,它将对象的创建和管理责任从代码中剥离出来,交由一个容器来处理。Spring框架是Java平台上的一个核心...

    Spring学习笔记&源码

    Spring框架是Java开发中不可或缺的一部分,它以其IoC(控制反转)和AOP(面向切面编程)的核心特性,极大地简化了企业级应用的开发。本资料“Spring学习笔记&源码”是基于网易云课堂黑马程序员的Spring四天精通课程...

    spring的学习笔记

    Spring框架是Java开发中广泛使用的轻量级框架,以其依赖注入(Dependency Injection,简称DI)和面向切面编程(Aspect-Oriented Programming,简称AOP)的核心特性,极大地简化了企业级应用的开发。以下是对Spring...

    狂神说Java-Spring学习笔记

    依赖注入是实现控制反转的一种常用方法。它允许对象在其生命周期中接收所需的依赖项,而不是自己创建或寻找这些依赖。 - **类型**:主要包括构造器注入、setter注入和字段注入三种方式。 - **优势**:提高了代码的...

    Spring学习笔记-cqupt

    在本篇 Spring 学习笔记中,我们将探讨 Spring 的入门、优点、组成以及重要的IOC理论。 1. **Spring 简介** Spring 是一个开源的、免费的 Java 框架,它的目标是减少企业级开发的复杂性。它集成了许多现有的技术,...

    Spring技术内幕 学习笔记

    首先,Spring的核心设计理念是依赖注入(Dependency Injection,简称DI),它通过反转控制权,使得组件之间的依赖关系由容器来管理,而不是由组件自身来管理。这一设计模式降低了组件之间的耦合度,提高了代码的可...

Global site tag (gtag.js) - Google Analytics