`

转载曹胜欢的细谈Spring(二)自己动手模拟spring

阅读更多
 

            在我们学习spring之前,根据spring的特性,我们来自己来模拟一个spring出来,也就是说不利用spring来实现spring的效果。本实例主要是实现springIOC功能。

首先我们把我们用的daoserviceentity定义出来:

 

Student.java :

 

  1. package com.bzu.entity;  
  2. public class Student {  
  3. private int id;  
  4. private String name;  
  5. private String address;  
  6. ******************set、get方法省略  
  7. }  
package com.bzu.entity;
public class Student {
private int id;
private String name;
private String address;
******************set、get方法省略
}


 

 

因为spring提倡的就是面向接口编程,所以在我们写dao层和service层具体实现之前,我们先定义接口,让我们的具体实现实现接口。接口的代码很简单,在这就不贴出来了。

 

  1. StudentdaoImp.java  
  2. public class StudentDaoImp implements StudentDao {  
  3. public void add(Student stu) {  
  4. System.out.println("stu is saved");  
  5. }  
  6. }  
StudentdaoImp.java
public class StudentDaoImp implements StudentDao {
public void add(Student stu) {
System.out.println("stu is saved");
}
}

 

 

 

 

StudentServiceImp.java

 

  1. public class StudentServiceImp implements StudentService {  
  2. StudentDao stuDao=null;  
  3. public StudentDao getStuDao() {  
  4. return stuDao;  
  5. }  
  6. public void setStuDao(StudentDao stuDao) {  
  7. this.stuDao = stuDao;  
  8. }  
  9. @Override  
  10. public void add(Student stu) {  
  11. stuDao.add(stu);  
  12. }  
  13. }  
public class StudentServiceImp implements StudentService {
StudentDao stuDao=null;
public StudentDao getStuDao() {
return stuDao;
}
public void setStuDao(StudentDao stuDao) {
this.stuDao = stuDao;
}
@Override
public void add(Student stu) {
stuDao.add(stu);
}
}

 

      这里要注意的是,我们这里是模拟spring,主要模拟spring中的IOC功能,所以在此我们一样要在service层中定义dao的实例,当然不用new出来,我们就通过springIOC把这里的dao层注入进来。不要忘了对dao提供setGet方法,因为IOC的底层其实就是利用反射机制实现的,他把dao注入进来,其实底层就是通过反射set进来的。

 

       好了,我们所需的dao层、service层还有entity定义好了之后,万事俱备只欠东风了,下一步我们就是定义我们自己的ClassPathXmlApplicationContext类了,通过他,在我们new出他的对象的时候,他来加载配置文件,然后把我们的dao操作注入到我们的service层,在spring中,ClassPathXmlApplicationContext类实现了BeanFactory接口,在此我们也定义一个BeanFactory接口,其实这个接口没什么具体的作用,我们就是为了来模拟spring。在定义这个接口和实现类之前,我们先来看一下我们所需的xml是怎么编写的,下面我们就具体来看一下beans.xml的配置:

 

Beans.xml

 

  1. <beans>  
  2. <bean id="stuDao" class="com.bzu.dao.imp.StudentDaoImp" />  
  3. <bean id="stuService" class="com.bzu.service.imp.StudentServiceImp" >  
  4. <property name="stuDao" bean="stuDao"/>  
  5. </bean>  
  6. </beans>  
<beans>
<bean id="stuDao" class="com.bzu.dao.imp.StudentDaoImp" />
<bean id="stuService" class="com.bzu.service.imp.StudentServiceImp" >
<property name="stuDao" bean="stuDao"/>
</bean>
</beans>


 

        看到这,相信大家都能感觉到这个配置文件太简单了,没有spring中那么多繁琐的配置,当然啦,我们这是只是实现其中的一个功能,spring提供了很多那么强大的功能,配置当然繁琐一些了。相信上边的代码不用我解释大家也能看懂了吧。

       好了,配置文件我们看完了,下一步我们一起来看一下我们的spring容器——ClassPathXmlApplicationContext具体是怎么实现的,我们首先还是来看一下他的接口定义:

BeanFactory.java:

 

 

  1. public interface BeanFactory {  
  2. public Object getBean(String id);  
  3. }  
public interface BeanFactory {
public Object getBean(String id);
}


 

我们看到,接口其实很简单,就定义了一个getBean方法,下面我们来看一下具体的实现类:

 

ClassPathXmlApplicationContext.java

 

 

  1. public class ClassPathXmlApplicationContext implements BeanFactory {  
  2. private Map<String, Object> beans = new HashMap<String, Object>();  
  3. public ClassPathXmlApplicationContext() throws Exception, Exception {  
  4. SAXBuilder sb = new SAXBuilder();  
  5. Document doc = sb.build(this.getClass().getClassLoader()  
  6. .getResourceAsStream("beans.xml")); // 构造文档对象   
  7. Element root = doc.getRootElement(); // 获取根元素HD   
  8. List list = root.getChildren("bean");// 取名字为disk的所有元素   
  9. for (int i = 0; i < list.size(); i++) {  
  10. Element element = (Element) list.get(i);  
  11. String id = element.getAttributeValue("id");  
  12. String clazz = element.getAttributeValue("class");  
  13. Object o = Class.forName(clazz).newInstance();  
  14. System.out.println(id);  
  15. System.out.println(clazz);  
  16. beans.put(id, o);  
  17. for (Element propertyElement : (List<Element>) element  
  18. .getChildren("property")) {  
  19. String name = propertyElement.getAttributeValue("name"); // userDAO   
  20. String bean = propertyElement.getAttributeValue("bean"); // u   
  21. Object beanObject = beans.get(bean);// UserDAOImpl instance   
  22. String methodName = "set" + name.substring(01).toUpperCase()  
  23. + name.substring(1);  
  24. System.out.println("method name = " + methodName);  
  25. Method m = o.getClass().getMethod(methodName,  
  26. beanObject.getClass().getInterfaces()[0]);  
  27. m.invoke(o, beanObject);  
  28. }  
  29. }  
  30. }  
  31. @Override  
  32. public Object getBean(String id) {  
  33. return beans.get(id);  
  34. }  
  35. }  
public class ClassPathXmlApplicationContext implements BeanFactory {
private Map<String, Object> beans = new HashMap<String, Object>();
public ClassPathXmlApplicationContext() throws Exception, Exception {
SAXBuilder sb = new SAXBuilder();
Document doc = sb.build(this.getClass().getClassLoader()
.getResourceAsStream("beans.xml")); // 构造文档对象
Element root = doc.getRootElement(); // 获取根元素HD
List list = root.getChildren("bean");// 取名字为disk的所有元素
for (int i = 0; i < list.size(); i++) {
Element element = (Element) list.get(i);
String id = element.getAttributeValue("id");
String clazz = element.getAttributeValue("class");
Object o = Class.forName(clazz).newInstance();
System.out.println(id);
System.out.println(clazz);
beans.put(id, o);
for (Element propertyElement : (List<Element>) element
.getChildren("property")) {
String name = propertyElement.getAttributeValue("name"); // userDAO
String bean = propertyElement.getAttributeValue("bean"); // u
Object beanObject = beans.get(bean);// UserDAOImpl instance
String methodName = "set" + name.substring(0, 1).toUpperCase()
+ name.substring(1);
System.out.println("method name = " + methodName);
Method m = o.getClass().getMethod(methodName,
beanObject.getClass().getInterfaces()[0]);
m.invoke(o, beanObject);
}
}
}
@Override
public Object getBean(String id) {
return beans.get(id);
}
}


 

代码贴出来了,不知道大家看懂没有。下面我来解释一下这段代码: 

      首先我们定义了一个容器Map<String, Object> beans,这个容器的作用就是用来装我们从配置文件里解析来的一个个bean,为什么要用map类型,我想大家也差不多能猜到吧,我们配置文件中每一个bean都有一个id来作为自己的唯一身份。我们把这个id存到mapkey里面,然后value就装我们的具体bean对象。说完这个容器之后,下面我们在来看一下ClassPathXmlApplicationContext的构造方法,这个构造方法是我们spring管理容器的核心,这个构造方法的前半部分是利用的jdom解析方式,把xml里面的bean一个个的解析出来,然后把解析出来的bean在放到我们bean容器里。如果这段代码看不懂的话,那你只好在去看看jdom解析xml了。好了,我们下面在来看一下这个构造的方法,后半部分主要是在对配置文件进行解析出bean的同时去查看一下这个bean中有没有需要注射bean的,如果有的话,他就去通过这些里面的property属性获取他要注射的bean名字,然后构造出set方法,然后通过反射,调用注入beanset方法,这样我们所需要的bean就被注入进来了。如果这段代码你看不懂的话,那你只能去看一下有关反射的知识了。最后我们就来看一下实现接口的getBean放了,其实这个方法很简单,就是根据提供的beanid,从bean容器内把对应的bean取出来。

 

好了,我们所需的东西都定义好了,下面我们据来测试一下,看看我们自己模仿的spring到底能不能自动把我们所需要的dao层给我们注入进来。

 

  1. public static void main(String[] args) throws Exception {  
  2. ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext();  
  3. Student stu = new Student();  
  4. StudentService service = (StudentService) context.getBean("stuService");  
  5. service.add(stu);  
  6. }  
public static void main(String[] args) throws Exception {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext();
Student stu = new Student();
StudentService service = (StudentService) context.getBean("stuService");
service.add(stu);
}

 

 

运行代码,控制台输出:

com.bzu.service.imp.StudentServiceImp

method name = setStuDao

stu is saved

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics