1. 程式人生 > >自己寫的一個迷你版spring IoC

自己寫的一個迷你版spring IoC


為了加強自己對spring IoC的理解,自己用jdom解析xml,利用反射機制寫了一個類似spring的程式。

spring最重要的一個思想之一就是IoC Inversion of Control(控制反轉)也教DI Dependency Injection(依賴注入),其實也不要管它叫什麼,我們是搞程式的,明白什麼意思就行了。

    下面看例子吧!舉例子最能說名問題了。假設我們寫的是一個使用者管理系統。首先有一個使用者例項為了簡單起見就寫了兩個屬性:

package com.hyh.pojo;

public class User {
	
	private String name;
	private String email;
	
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public String getEmail() {
		return email;
	}
	public void setEmail(String email) {
		this.email = email;
	}
	
}
接下來寫Dao層,為了能跨資料庫平臺我們在這裡寫了一個UserDao介面,我們就寫一個add方法。
package com.hyh.dao;

import com.hyh.pojo.User;

public interface UserDao {
	
	public void add(User u);

}

下面是UserDao的實現類:
package com.hyh.dao.impl;

import com.hyh.dao.UserDao;
import com.hyh.pojo.User;

public class UserDaoImpl implements UserDao{

	@Override
	public void add(User u) {
                //這裡我們就不去連線資料庫了,列印一句話。
		System.out.println("a user saved!");
	}

}
再接下來就是service層了:
package com.hyh.service;

import com.hyh.dao.UserDao;
import com.hyh.pojo.User;

public class UserService {
	
	//如果在這裡直接建立userDao,你會發現UserService和UserDao的耦合性太強了。
	//我們就想辦法不讓UserService建立UserDao,而是依賴容器注入,這裡我們就依賴我們自己寫的程式注入。這樣在很大程度上降低了耦合度
	private UserDao userDao ;//= new UserDaoImpl();
	
	public UserDao getUserDao() {
		return userDao;
	}

	public void setUserDao(UserDao userDao) {
		this.userDao = userDao;
	}

	public void save(User u){
		
		userDao.add(u);
	}
	

}

在接下來我們就應該想辦法怎樣讓程式注入所需要的Bean了。所以我們建立一個BeanFactory用於產生各種各樣的bean:
package com.hyh.test;

public interface BeanFactory {
	//根據Id得到一個Bean,
	public Object getBean(String id);
}
BeanFactory有了,我們就要想辦法去建立bean了,所以我們需要建立一個類實現BeanFactory介面來產生Bean,這裡我為了模擬spring所以我建立了ClassPathXmlApplicationContext:
package com.hyh.test;


import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.jdom.Document;
import org.jdom.Element;
import org.jdom.input.SAXBuilder;
//用到的包自己新增
public class ClassPathXmlApplicationContext implements BeanFactory {
	
	private Map<String , Object> beans = new HashMap<String, Object>();
	
	
	//這裡我用的是jdom解析xml檔案。解析xml檔案有好多種,如果有興趣,看以參考這篇文章。
	public ClassPathXmlApplicationContext() throws Exception {
		SAXBuilder sb=new SAXBuilder();
	    
	    Document doc=sb.build(this.getClass().getClassLoader().getResourceAsStream("beans.xml")); 
	    Element root=doc.getRootElement(); 
	    List list=root.getChildren("bean");
	    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"); 
	    	   String bean = propertyElement.getAttributeValue("bean"); 
	    	   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);
	       }
	       
	       
	    }  
	  
	}


        //這裡根據傳進來的id得到一個bean物件
	public Object getBean(String id) {
		return beans.get(id);
	}

}
我們來看看我們在xml裡面怎麼配置吧!
<beans>
	<bean id="userDao" class="com.hyh.dao.impl.UserDaoImpl" />
	
	<!--這裡我們模擬了spring的自動裝配-->
	<bean id="userService" class="com.hyh.service.UserService" >
		<property name="userDao" bean="userDao"/>
	</bean>
</beans>

下面我們來測試一下吧!測試用的是junit4自己新增所需要的jar檔案:
package com.hyh.service;

import org.junit.Test;

import com.hyh.pojo.User;
import com.hyh.test.BeanFactory;
import com.hyh.test.ClassPathXmlApplicationContext;

public class TesstUserService {
	
	private UserService userService;// = new UserService();
	
	
	@Test
	public void testAdd() throws Exception{
		
		BeanFactory bf = new ClassPathXmlApplicationContext();
               //這裡我們根據我們在配置檔案裡寫的userService來得到UserServiced物件。
                userService = (UserService) bf.getBean("userService");
		
		System.out.println(bf.getClass().getName());
		System.out.println(userService.getClass().getSimpleName());
		
		User u = new User();
		
		userService.save(u);
	}

}
執行testAdd方法,你會看到 a user saved!!!

這就是spring IoC的基本原理。

本文參考:

         馬士兵視訊