`

spring注入原理

阅读更多

      IOC(Inverse of Control)可翻译为“控制反转”,但大多数人都习惯将它称为“依赖注入”。在Spring中,通过IOC可以将实现类 、参数信息等配置在其对应的配置文件中 ,那么当需要更改实现类或参数信息时,只需要修改配置文件即可,这种方法在上例的基础上更进一步的降低了类与类之间的耦合。我们还可以对某对象所需要的其它对象进行注入 ,这种注入都是在配置文件中做的,Spring的IOC的实现原理利用的就是Java的反射机制,Spring还充当了工厂的角色,我们不需要自己建立工厂类 。Spring的工厂类会帮我们完成配置文件的读取、利用反射机制注入对象等工作,我们可以通过bean的名称获取对应的对象。下面让我们看看如下的模拟Spring的bean工厂类:

1、BeanFactory.java

package com.yt.manager.spring;

import java.io.InputStream;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import org.apache.log4j.Logger;
import org.dom4j.Attribute;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;

/**
 * @ClassName: BeanFactory
 * @Project: base-info
 * @Author: zxf
 * @Date: 2011-5-19
 */
public class BeanFactory {
	
	Logger log = Logger.getLogger(BeanFactory.class);
	private Map<String, Object> beanMap = new HashMap<String, Object>();

	/**
	 * bean工厂的初始化
	 * 
	 * @param xml
	 */
	public void init(String xml) {
		try {
			//读取指定的配置文件
			SAXReader reader = new SAXReader();
			//从class目录下获取指定的配置文件
			ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
			InputStream inputStream = classLoader.getResourceAsStream(xml);
			//读取xml文件
			Document document = reader.read(inputStream);
			//获取跟节点
			Element root = document.getRootElement();
			//遍历bean节点
			Element foo;
			for(Iterator iteBean = root.elementIterator("bean");iteBean.hasNext();){
				foo = (Element)iteBean.next();
				//获取bean的属性id和class
				Attribute id = foo.attribute("id");
				Attribute cls = foo.attribute("class");
				//利用java反射机制,通过class的名称获取Class对象
				Class bean = Class.forName(cls.getText());
				//获取对应class信息
				java.beans.BeanInfo info = java.beans.Introspector.getBeanInfo(bean);
				//获取其属性描述
				java.beans.PropertyDescriptor pd[] = info.getPropertyDescriptors();
				//设置值的方法
                Method mSet = null;
                //创建一个对象(创建此 Class 对象所表示的类的一个新实例。如同用一个带有一个空参数列表的 new 表达式实例化该类。如果该类尚未初始化,则初始化这个类。)
                Object obj = bean.newInstance();
                
                //遍历该bean的property属性
                for(Iterator iteProperty = foo.elementIterator("property");iteProperty.hasNext();){
                	Element elementProperty = (Element)iteProperty.next();
                	//获取该property的name属性
                	Attribute name = elementProperty.attribute("name");
                	String value = null;
                	//获取该property的子元素value的值
                	for(Iterator iteValue = elementProperty.elementIterator("value");iteValue.hasNext();){
                		Element elementValue = (Element)iteValue.next();
                		value = elementValue.getText();
                		break;
                	}
                	
                	for (int k = 0; k < pd.length; k++) {
                		log.info(pd[k].getName());
                        if (pd[k].getName().equalsIgnoreCase(name.getText())) {
                               mSet = pd[k].getWriteMethod();
                               //利用Java的反射机制调用对象的某个set方法,并将值设置进去
                               mSet.invoke(obj, value);
                        }
                 }
                }
               //将对象放入beanMap中,其中key为id值,value为对象
                beanMap.put(id.getText(), obj);
			}

		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	
    /**
     * 通过bean的id获取bean的对象.
     * @param beanName bean的id
     * @return 返回对应对象
     */
     public Object getBean(String beanName) {
           Object obj = beanMap.get(beanName);
           return obj;
     }

     public static void main(String[] args) {
         BeanFactory factory = new BeanFactory();
         factory.init("config.xml");
         JavaBean javaBean = (JavaBean) factory.getBean("javaBean");
         System.out.println("userName=" + javaBean.getUserName());
         System.out.println("password=" + javaBean.getPassWord());
     }
}

 2、JavaBean.java

package com.yt.manager.spring;

/**
 * @Description:
 * @ClassName: JavaBean
 * @Project: base-info
 * @Author: zxf
 * @Date: 2011-5-19
 */
public class JavaBean {
	private String userName;
	private String passWord;

	public String getUserName() {
		return userName;
	}

	public void setUserName(String userName) {
		this.userName = userName;
	}

	public String getPassWord() {
		return passWord;
	}

	public void setPassWord(String passWord) {
		this.passWord = passWord;
	}

}

 3、config.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans>
	<bean id = "javaBean" class="com.yt.manager.spring.JavaBean">
		<property name="userName">
			<value>这是姓名</value>
		</property>
		<property name="passWord">
			<value>这是密码</value>
		</property>
	</bean>
</beans>

      可以看到,虽然在main()方法中没有对属性赋值,但属性值已经被注入,在BeanFactory类中的Class bean = Class.forName(cls.getText()); 通过类名来获取对应的类,mSet.invoke(obj, value);通过invoke方法来调用特定对象的特定方法,实现的原理都是基于Java的反射机制,在此我们有一次见证了Java反射机制的强大。当然,这只是对IOC的一个简单演示,在Spring中,情况要复杂得多,例如,可以一个bean引用另一个bean,还可以有多个配置文件、通过多种方式载入配置文件等等。不过原理还是采用Java的反射机制来实现IOC的。
       在本文中,笔者通过讲述Java反射机制概述与初探、IOC使用的背景、IOC粉墨登场等内容,演示了Java反射机制API的强大功能,并通过编写自己的简单的IOC框架,让读者更好的理解了IOC的实现原理。本文通过IOC的一个简要实现实例,模拟了Spring中IOC的实现,虽然只是完成了Spring中依赖注入的一小部分工作, 但是很好的展现了Java反射机制在Spring中的应用,能使我们能更好的从原理上了解IOC的实现,也能为我们实现自己的准Spring框架提供方案,有兴趣的朋友可以通过Spring的源码进行IOC的进一步的学习。

 

 附:通过java反射机制获取指定类的属性和方法

package com.yt.manager.spring;

import java.beans.BeanInfo;
import java.beans.Introspector;
import java.beans.MethodDescriptor;
import java.beans.PropertyDescriptor;

/**
 * @Description:java.beans包的使用
 * @ClassName: JavaBeans
 * @Project: base-info
 * @Author: zxf
 * @Date: 2011-5-19
 */
public class JavaBeans {

	/**
	 * @param args
	 * @throws Exception
	 */
	@SuppressWarnings("rawtypes")
	public static void main(String[] args) throws Exception {
		// 返回与带有给定字符串名的类或接口相关联的 Class 对象
		Class bean = Class.forName("com.yt.manager.spring.JavaBean");
		// 获取指定类的信息
		BeanInfo info = Introspector.getBeanInfo(bean);
		// 遍历指定类的方法
		MethodDescriptor[] methods = info.getMethodDescriptors();
		for (int i = 0; i < methods.length; i++) {
			System.out.println("方法:"+methods[i].getDisplayName());
		}
		// 遍历指定类的属性
		PropertyDescriptor[] propertys = info.getPropertyDescriptors();
		for (int j = 0; j < propertys.length; j++) {
			System.out.println("属性:"+propertys[j].getName());
		}
	}

}
 
package com.yt.manager.spring;

import java.lang.reflect.Method;

/**
 * @Description: java.lang.reflect.method类中invoke方法的使用
 * @ClassName: MethodInvoke
 * @Project: base-info
 * @Author: zxf
 * @Date: 2011-5-19
 */
public class MethodInvoke {

	/** 
	 * @param args
	 * @throws ClassNotFoundException
	 */
	public static void main(String[] args) throws Exception {
		
		JavaBean javaBean = new JavaBean();
		//获取指定类的指定方法,
		Class c = Class.forName("com.yt.manager.spring.JavaBean");
		Method method = c.getMethod("setUserName", new Class[] { String.class });
		//对带有指定参数的指定对象调用由此 Method 对象表示的底层方法,调用对象javaBean的setuserName方法,参数为"testName"
		method.invoke(javaBean, "testName");
		
		System.out.println(javaBean.getUserName());
	}

}
 

 

 

 

 

 

 

 

 

 

 

分享到:
评论
1 楼 jackson200 2018-03-23  
讲解的狠不错!谢谢!

相关推荐

    spring 注入原理

    ### Spring注入原理详解 在Java开发领域,Spring框架无疑占据着举足轻重的地位,尤其在企业级应用中,Spring的依赖注入(Dependency Injection,DI)特性极大地简化了对象之间的依赖管理,使得代码更加模块化、可...

    spring依赖注入的实现原理

    Spring依赖注入(Dependency Injection,简称DI)是Java应用开发中常用的设计模式,它极大地提高了代码的可测试性和可维护性。在Spring框架中,依赖注入是核心特性之一,通过控制反转(Inversion of Control,IoC)...

    SQL注入原理以及Spring Boot如何防止SQL注入(含详细示例代码)

    SQL注入是一种严重的安全威胁,它...综上所述,理解SQL注入的原理并采取相应的防护措施是确保应用程序安全的关键。Spring Boot通过提供JdbcTemplate和Spring Data JPA等工具,为开发者提供了防止SQL注入的有效手段。

    springIoc实现原理

    **Spring Ioc 实现原理详解** Spring Ioc(Inversion of Control,控制反转)是Spring框架的核心特性之一,它改变了传统应用程序中对象的创建和管理方式。在传统的软件设计中,对象的创建和依赖关系的维护通常由...

    Spring学习笔记(6)----编码剖析Spring依赖注入的原理

    本篇学习笔记将深入剖析Spring依赖注入的原理,通过源码分析帮助我们理解这一核心机制。 首先,依赖注入允许我们解耦组件之间的关系,使得各个组件可以独立地进行开发、测试和维护。在Spring中,DI主要通过两种方式...

    Spring依赖注入原理解析.doc

    Spring依赖注入机制,也被称为控制反转(Inversion of Control,IOC),是Spring框架的核心特性,它使得应用程序的组件之间能够实现松散耦合。在传统编程中,对象通常自行创建和管理它们所依赖的其他对象,而在...

    Spring依赖注入检查.

    本文将深入探讨Spring依赖注入的概念、工作原理以及如何在实际项目中应用。 依赖注入(Dependency Injection,简称DI)是一种设计模式,它允许组件之间通过外部源来管理其依赖关系,而不是由组件自己来创建或查找...

    关于spring boot中几种注入方法的一些个人看法

    @Autowired 的工作原理是,在启动 Spring IoC 时,容器自动装载了一个 AutowiredAnnotationBeanPostProcessor 后置处理器,当容器扫描到 @Autowired、@Resource 或 @Inject 时,就会在 IoC 容器自动查找需要的 bean...

    Spring技术内幕:深入解析Spring架构与设计原理.pdf

    Spring技术内幕:深入解析Spring架构与设计原理 Spring技术内幕 Spring是一个基于Java的开源框架,旨在简化Java企业应用的开发。Spring的目标是提供一个简洁、灵活、可扩展的框架,以帮助开发者快速构建企业级...

    Spring读取配置文件原理(Spring如何依赖注入的)

    本文将深入探讨Spring如何通过读取配置文件实现依赖注入,并讲解相关源码,帮助理解其工作原理。 在Spring中,配置文件通常为XML格式,如`applicationContext.xml`,它定义了bean的实例化、属性设置、装配关系等。...

    浅谈Spring IoC容器的依赖注入原理

    浅谈Spring IoC容器的依赖注入原理 Spring IoC容器的依赖注入原理是Spring框架的核心机制之一,负责将服务对象(Bean)实例化并将其提供给客户端使用。依赖注入原理可以分为两个阶段:IoC容器初始化和Bean实例化。 ...

    spring依赖注入三种方式 测试源码

    本文将深入探讨Spring中的三种依赖注入方式,并结合源码分析其工作原理。 1. **构造器注入** 构造器注入是通过构造函数来传递依赖对象的。当创建一个类的新实例时,Spring容器会调用带有所需参数的构造函数,从而...

    编码剖析Spring依赖注入的原理

    本文将深入解析Spring中的依赖注入原理,帮助开发者更好地理解和应用这一核心功能。 依赖注入(Dependency Injection,简称DI)是Spring框架的核心之一,它允许组件之间的依赖关系在运行时由外部容器来管理,而不是...

    Spring技术内幕:深入解析Spring架构与设计原理[汇编].pdf

    * 依赖注入(Dependency Injection): Spring使用依赖注入来解耦合应用程序中的组件,提高系统的灵活性和可维护性。 * AOP(Aspect-Oriented Programming): Spring使用AOP来实现面向方面的编程,提高系统的可维护...

    Spring技术内幕:深入解析Spring架构与设计原理(第2版) .pdf

    《Spring技术内幕:深入解析Spring架构与设计原理(第2版)》这本书主要聚焦于Spring框架的核心架构和技术细节,帮助读者全面理解Spring的工作机制、设计理念以及实现方式。下面将根据书名及其描述来展开相关知识点。 ...

    Spring技术内幕:深入解析Spring架构与设计原理

    《Spring技术内幕:深入解析Spring架构与设计原理》是一本深度剖析Spring框架核心机制与设计理念的专业书籍。本书针对Java开发者,特别是对Spring框架有深入学习需求的读者,提供了丰富的技术细节和实践指导。 ...

    对Spring中注解怎么实现的一些基本原理

    在Spring框架中,注解是实现依赖注入和配置的核心机制之一。本文将深入探讨Spring注解的基本原理,包括它们如何被解析、处理以及如何影响应用程序的生命周期。 首先,我们需要了解注解在Java语言中的本质。注解是一...

Global site tag (gtag.js) - Google Analytics