(5) 编码剖析Spring装配基本属性的原理【附加:注入依赖对象的两种方式】

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">
           
           <bean id="personDao" class="cn.itm.dao.impl.PersonDaoBean"></bean>
           
           <bean id="personService" class="cn.itm.service.impl.PersonServiceBean" >
           		<!--  实现 注入 -->
           		<property name="personDao" ref="personDao"></property>
           		
           		<property name="name" value="itm"></property>
           		<property name="id" value="88"></property>
           </bean>
           
    
</beans>


下面包含:属性的name,id.

package cn.itm.service.impl;

import cn.itm.dao.PersonDao;
import cn.itm.service.PersonService;

public class PersonServiceBean implements PersonService{

	// 使用 Set方法 是实现依赖注入:
	private PersonDao personDao;
	
	// 注入 属性:<property name="name" value="itm"></property>
	private String name;
	
	private Integer id;
	
	public Integer getId() {
		return id;
	}


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


	public String getName() {
		return name;
	}


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


	public void setPersonDao(PersonDao personDao) {
		this.personDao = personDao;
	}


	public void save(){
		
		System.out.println("id = " + id + "," +name);
		// 调用 依赖对象注入进来的方法了。
		personDao.add();
	}
	
}

public interface PersonService {

	public abstract void save();

}

package cn.itm.dao.impl;

import cn.itm.dao.PersonDao;


public class PersonDaoBean implements PersonDao {

	
	public void add(){
		System.out.println("执行PersonDaoBean的add方法。。。");
	}
}

package cn.itm.dao;

public interface PersonDao {

	public abstract void add();

}


装载基本属性的value


package junit.test;

/**
 * 因为 Property是 属于 bean的属性,所以也要在  bean中定义Property。
 * 
 * @author Administrator
 *
 */
public class PropertyDefinition {

	private  String name;
	private String ref;
	
	private String value;
	public String getValue() {
		return value;
	}
	public void setValue(String value) {
		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 PropertyDefinition(String name, String ref,String value) {
		this.name = name;
		this.ref = ref;
		this.value = value;
	}
	
	
}

Spring容器  bean:

package junit.test;

import java.util.ArrayList;
import java.util.List;

public class BeanDefinition {

	private String id;
	private String className;
	
	private List<PropertyDefinition> propertys = new ArrayList<PropertyDefinition>();
	
	
	public List<PropertyDefinition> getPropertys() {
		return propertys;
	}
	public void setPropertys(List<PropertyDefinition> propertys) {
		this.propertys = propertys;
	}
	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 BeanDefinition(String id, String className) {
		this.id = id;
		this.className = className;
	}
	
	
}

下面是:模拟Spring容器装配基本属性的原理



代码如下:

package junit.test;


import java.beans.Introspector;
import java.beans.PropertyDescriptor;
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.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.XPath;
import org.dom4j.io.SAXReader;

public class ItmClassPathXMLApplicationContext {
	
	private List<BeanDefinition> beanDefines = new ArrayList<BeanDefinition>();
	// 存放实例
	private Map<String,Object> sigletons = new HashMap<String,Object>();
	
	public ItmClassPathXMLApplicationContext(String fileName){
		this.readXML(fileName);
		this.instanceBeans();
		this.injectObject();
	}

	private void injectObject() {
		for(BeanDefinition beanDefinition : beanDefines){
			// 得到  bean 。。
			Object bean = sigletons.get(beanDefinition.getId());
			if(bean != null){
				try {
					// 得到 bean的属性描述:
					PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();
					// 循环  bean里面的 所有的属性:
					for( PropertyDefinition propertyDefinition: beanDefinition.getPropertys()){
						for(PropertyDescriptor propertyDesc /*这里是 bean 里面的属性*/ : ps){
							if(propertyDefinition.getName().equals(propertyDesc.getName())){
								// 如果相等 说明是存在 于 这个bean的。。。
								Method setter = propertyDesc.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(), propertyDesc.getPropertyType());
									}
									setter.setAccessible(true); // 允许访问 私有的方法。。
									setter.invoke(bean, value);// 把引用对象注入到属性。
								}
								break;
							}
						}
					}
				} catch (Exception e) {
					e.printStackTrace();
				}
			}
		}
	}

	/**
	 * 通过反射技术,完成  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
	 */
	@SuppressWarnings("unchecked")
	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"); // 加入命名空间
			
			// 创建beans/bean 查询路径。
			XPath xsub = document.createXPath("//ns:beans/ns:bean");  
			
			// 设置命名空间。
			xsub.setNamespaceURIs(nsMap);  								  
			
			// 获取文档下 所有bean节点:
			List<Element> beans = xsub.selectNodes(document);
			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");
					
					System.out.println(propertyName + "==" + propertyRef);
					PropertyDefinition propertyDefinition = new PropertyDefinition(propertyName, propertyRef, propertyValue);
					
					// 放到  bean里面去:
					beanDefine.getPropertys().add(propertyDefinition);
					
				}
				beanDefines.add(beanDefine);
			}
			
		}catch(Exception e){
			e.printStackTrace();
		}
	}

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

测试类:

package junit.test;


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

import cn.itm.service.PersonService;

public class SpringTest {

	@BeforeClass
	public static void setUpBeforeClass() throws Exception {
	}

	// 专门用来实例化  Spring 容器的。
	@Test public void instanceSpring(){
		
//		ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");
		ItmClassPathXMLApplicationContext ctx = new ItmClassPathXMLApplicationContext("beans.xml");
		PersonService personService = (PersonService) ctx.getBean("personService");
		personService.save();
	
	}
}

ok,没有问题。




本文源自:自学传智播客黎活明老师的视频,消化吸收总结而至。


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值