上一篇文章介绍了Java反射机制在Spring IOC中的应用,知道了BeanFactory底层的实现原理。
原理搞懂了,对Spring IOC理解起来也很容易。
先来看看Java代码获取Spring中Bean的代码(一共有五种方式,这里只展示其中一种方法):
有没有发现上面的代码与利用反射实现工厂模式的代码很相似。对,你没有看错,Spring中的BeanFactory用到的就是简单工厂模式。
现在的思路就更加清晰了,要想实现Spring中的BeanFactory,无非就用到了以下几个技术:
1.使用简单工厂模式来处理bean容器。
2.解析xml文件,获取配置中的元素信息。
3.利用反射获实例化配置信息中的对象。
4.如果有对象注入,使用invoke()方法。
5.实例化的对象放入bean容器中,并提供getBean方法。
通过以上步骤就实现了spring的BeanFactory功能,只要在配置文件中配置好,实例化对象的事情交给BeanFactory来实现,用户不需要通过new对象的方式实例化对象,直接调用getBean方法即获取对象实例。
具体实现代码:
新建一个xml文件,内容如下:
<?xml version="1.0" encoding="UTF-8"?>
<beans>
<bean id="courseDao" class="com.qcjy.learning.Dao.impl.CourseDaoImpl"> <bean id="courseService" class="com.qcjy.learning.service.impl.CourseServiceImpl">
<!-- 控制调用setCourseDao()方法,将容器中的courseDao bean作为传入参数 -->
<property name="courseDao" ref="courseDao"></property>
</bean> </beans>
接下来实现BeanFactory工厂,提供init方法,和getBean方法,在init方法中解析xml,利用反射实例话对象,存入bean容器中,代码如下:
import java.beans.BeanInfo;
import java.beans.PropertyDescriptor;
import java.io.InputStream;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import org.dom4j.Attribute;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
public class BeanFactory { //bean容器
private Map<String, Object> contianer = new HashMap<String, Object>(); /**
* <p>Discription:bean工厂的初始化</p>
* @param xml xml配置文件路径
* @author : lcma
* @update : 2016年9月20日上午9:04:41
*/
public void init(String xml) {
try {
// 读取指定的配置文件
SAXReader reader = new SAXReader();
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
// 从class目录下获取指定的xml文件
InputStream ins = classLoader.getResourceAsStream(xml);
Document doc = reader.read(ins);
Element root = doc.getRootElement();
Element foo;
// 遍历bean
for (Iterator i = root.elementIterator("bean"); i.hasNext();) {
foo = (Element) i.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;
// 创建一个对象
Object obj = bean.newInstance();
// 遍历该bean的property属性
for (Iterator ite = foo.elementIterator("property"); ite.hasNext();) {
Element foo2 = (Element) ite.next();
// 获取该property的name属性
Attribute name = foo2.attribute("name");
String value = null;
// 获取该property的子元素value的值
for (Iterator ite1 = foo2.elementIterator("value"); ite1.hasNext();) {
Element node = (Element) ite1.next();
value = node.getText();
break;
}
for (int k = 0; k < pd.length; k++) {
if (pd[k].getName().equalsIgnoreCase(name.getText())) {
mSet = pd[k].getWriteMethod();
// 利用Java的反射机制调用对象的某个set方法,并将值设进去
mSet.invoke(obj, value);
}
}
}
// 将对象放入beanMap中,其中key为id值,value为对象
contianer.put(id.getText(), obj);
}
} catch (Exception e) {
System.out.println(e.toString());
}
} /**
* <p>Discription:通过bean的id在容器中获取bean对象</p>
* @param beanName bean的唯一标识id
* @return
* @author : lcma
* @update : 2016年9月20日上午9:05:00
*/
public Object getBean(String beanName) {
Object obj = contianer.get(beanName);
return obj;
} }
测试方法:
/**
* <p>Discription:测试方法</p>
* @param args
* @author : lcma
* @update : 2016年9月20日上午9:06:06
*/
public static void main(String[] args) {
//实例化BeanFactory
BeanFactory factory = new BeanFactory();
//调用初始化方法,传入xml路径
factory.init("spring.xml");
//通过bean id 获取对象
CourseService courseService = (CourseService) factory.getBean("courseService");
//调用对象方法
courseService.findAll();
}
还要提供CourseService和CourseDao两个接口及实现类,这里就不提供了。
上面的代码已经简单的模拟实现了BeanFactory的功能啦,Spring框架里面的代码要比我们这个复杂的多,因为要考虑到安全性、稳定性、异常等等因素,但是原理都一样。
https://blog.****.net/mlc1218559742/article/details/52776160