上一篇文章介绍了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.csdn.net/mlc1218559742/article/details/52776160