自定义实现spring式的IOC

这两天想好好学习一下spring,对于spring的东西无非就是一些配置,但是不想只停留在应用的层面。一直以来对spring的内部的运行的原理非常好奇,想着研究一下源码,但是把源码拿过来如大海捞针,实在是无从下手,于是乎,自己综合查了的一些资料实现了spring式的IOC,感觉很有意思。下边把代码贴出来:

首先是定义了两个测试用的接口:

package com.ioc.dao;

public interface Car {
	
	void cleanCar();
	
}

package com.ioc.dao;

public interface CarService {
	void sell();
}

和两个接口的实现:

package com.ioc.dao.impl;

import com.ioc.dao.Car;

public class CarImpl implements Car {

	@Override
	public void cleanCar() {
		System.out.println("正在进行洗车!");
	}

}

package com.ioc.dao.impl;

import com.ioc.dao.Car;
import com.ioc.dao.CarService;

public class CarServiceImpl implements CarService {

	private Car car;
	
	public Car getCar() {
		return car;
	}
	public void setCar(Car car) {
		this.car = car;
	}
	
	@Override
	public void sell() {
		car.cleanCar();
	}
}

以上四个类或者是接口用于测试,主要是验证依赖注入


BeanObject这个类用于处理xml文件中bean的对象,List集合用于接收属性

package com.ioc.realize;

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

/**
 * 对应于xml文件中的
 * <bean id="car" class="com.ioc.dao.impl.CarImpl"></bean>  
 * Copyright 2015年10月27日
 * created by txxs
 * all right reserved
 */
public class BeanObject {
	
	private String id;
	private String className;
	//属性集合
	private List<AttributeObject> propertys = new ArrayList<AttributeObject>();
	
	public BeanObject(String id, String className) {
		this.id = id;
		this.className = className;
	}
	
	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 List<AttributeObject> getPropertys() {
		return propertys;
	}
	public void setPropertys(List<AttributeObject> propertys) {
		this.propertys = propertys;
	}
	
}

AttributeObject这个类用来处理XML中的属性

package com.ioc.realize;

/**
 * 存放属性的对象,对应
 *  <property name="car" ref="car"></property>
 * Copyright 2015年10月27日
 * created by txxs
 * all right reserved
 */
public class AttributeObject {
	
	private String name;
	private String ref;
	
	public AttributeObject(String name, String ref) {
		this.name = name;
		this.ref = ref;
	}
	
	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;
	}

}

下边就是自己的上下文SelfClassPathXMLApplicationContext,首先读取XML文件,读取完成后建立bean和属性的集合,然后实例化bean,再将依赖的bean注入

package com.ioc.realize;

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

public class SelfClassPathXMLApplicationContext {
	
	private List<BeanObject> beanObjects = new ArrayList<BeanObject>();  
    private Map<String, Object> sigletons = new HashMap<String, Object>();  
      
    public SelfClassPathXMLApplicationContext(String filename){  
        this.readXML(filename);  
        this.instanceBeans();  
        this.injectObject();  
    }  
    /** 
     * 为bean对象的属性注入值 
     */  
    private void injectObject() {  
        for(BeanObject beanObject : beanObjects){  
            Object bean = sigletons.get(beanObject.getId());  
            if(bean!=null){  
                try {  
                    PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();  
                    for(AttributeObject property : beanObject.getPropertys()){  
                        for(PropertyDescriptor properdesc : ps){  
                            if(property.getName().equals(properdesc.getName())){  
                            	Method setter = properdesc.getWriteMethod();//获取属性的setter方法   
                                if(setter!=null){  
                                    Object value = sigletons.get(property.getRef());  
                                    setter.setAccessible(true);  
                                    setter.invoke(bean, value);//把引用对象注入到属性  
                                }  
                                break;  
                            }  
                        }  
                    }  
                } catch (Exception e) {  
                }  
            }  
        }  
    }  
    /** 
     * 完成bean的实例化 
     */  
    private void instanceBeans() {  
        for(BeanObject beanObject : beanObjects){  
            try {  
            	String aa = beanObject.getClassName().trim();
                if(beanObject.getClassName()!=null && !"".equals(beanObject.getClassName().trim()))
                	sigletons.put(beanObject.getId(),Class.forName(beanObject.getClassName()).newInstance());  
            } catch (Exception e) {  
                e.printStackTrace();  
            }  
        }  
          
    }  
    /** 
     * 读取xml配置文件 
     * @param filename 
     */  
    private void readXML(String filename) {
    	//创建xml解析对象
    	SAXReader saxReader = new SAXReader();
        Document document=null;     
        try{  
           //filename不能以'/'开头时,path是从ClassPath根下获取
           URL xmlpath = this.getClass().getClassLoader().getResource(filename);  
           //读取XML文件,获得document对象
           document = saxReader.read(xmlpath);  
           Map<String,String> nsMap = new HashMap<String,String>(); 
           //加入命名空间
           nsMap.put("ns","http://www.yours.org/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){  
        	   	//获取id属性值  
        	   	String id = element.attributeValue("id");
        	   	//获取class属性值   
                String clazz = element.attributeValue("class");        
                BeanObject beanObject = new BeanObject(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");  
                    AttributeObject attributeObject = new AttributeObject(propertyName, propertyref);  
                    beanObject.getPropertys().add(attributeObject);  
                }  
                beanObjects.add(beanObject);  
             }   
            }catch(Exception e){     
                e.printStackTrace();  
            }  
    }  
    /** 
     * 获取bean实例 
     * @param beanName 
     * @return 
     */  
    public Object getBean(String beanName){  
        return this.sigletons.get(beanName);  
    }  
}

测试类:

package com.ioc.test;

import com.ioc.dao.CarService;
import com.ioc.realize.SelfClassPathXMLApplicationContext;

public class SpringIocTest {
	 
	 public static void main(String args[]){
		 SelfClassPathXMLApplicationContext ctx = new SelfClassPathXMLApplicationContext("IocConfig.xml");  
		 CarService carService = (CarService)ctx.getBean("carService");  
		 carService.sell();
	 }
}

输出结果:

正在进行洗车!

下边是一些资料:

http://uule.iteye.com/blog/2059920

http://dingjun1.iteye.com/blog/210165

http://lgbolgger.iteye.com/category/322988

http://www.iteye.com/topic/1125248


  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: 很高兴回答您的问题。首先,需要定义一个Bean类,然后使用反射机制创建Bean实例并将其存储在Map中。接着,需要解析Bean的依赖关系并进行注入。最后,需要实现Bean的生命周期管理,包括初始化和销毁。具体实现可以参考Spring源码。 ### 回答2: 实现一个简单的自定义Spring IOC容器需要以下步骤: 1. 定义一个bean类,该类包含需要注入的属性和对应的setter方法。 2. 创建一个容器类,用于管理bean的实例。在该容器类中,定义一个Map用来存储bean的名称和对应的实例。 3. 创建一个注入类,该类负责根据配置文件或注解来实例化和注入bean。可以使用反射机制来动态创建对象。 4. 编写一个配置类,该类用于读取配置文件或注解,获取需要创建的bean的信息。 5. 在容器类中,编写一个方法用于注册bean实例,将bean的名称和对应的实例放入Map中。 6. 在容器类中,编写一个方法用于从容器中获取bean的实例。 7. 将以上步骤整合到一个主程序中,测试容器类的功能。 以下是一个简单的实现示例: 1. 创建一个User类,包含name和age属性,并提供相应的setter方法。 ```java public class User { private String name; private int age; // Getter and Setter methods // ... } ``` 2. 创建一个容器类MyIOCContainer,使用Map来存储bean的名称和对应的实例。 ```java public class MyIOCContainer { private Map<String, Object> beans = new HashMap<>(); public void registerBean(String name, Object bean) { beans.put(name, bean); } public Object getBean(String name) { return beans.get(name); } } ``` 3. 创建一个注入类,用于实例化和注入bean。 ```java public class BeanInjector { public static void inject(Object bean) { // 通过反射机制实例化对象 // 通过反射机制注入属性 } } ``` 4. 创建一个配置类,使用注解来获取bean的信息。 ```java public class AppConfig { @Bean public User getUser() { User user = new User(); // 设置属性值 return user; } } ``` 5. 在容器类中实现注册和获取bean的方法。 ```java public class MyIOCContainer { // ... public void registerBean(String name, Object bean) { BeanInjector.inject(bean); // 注入属性 beans.put(name, bean); } public Object getBean(String name) { return beans.get(name); } // ... } ``` 6. 编写一个主程序,测试容器类的功能。 ```java public class Main { public static void main(String[] args) { MyIOCContainer container = new MyIOCContainer(); // 创建配置类实例 AppConfig config = new AppConfig(); // 获取bean的信息 Method[] methods = config.getClass().getMethods(); for (Method method : methods) { if (method.isAnnotationPresent(Bean.class)) { try { // 调用配置类的方法获取bean实例 Object bean = method.invoke(config); // 注册到容器中 container.registerBean(method.getName(), bean); } catch (Exception e) { e.printStackTrace(); } } } // 从容器中获取bean的实例 User user = (User) container.getBean("getUser"); System.out.println("Name: " + user.getName()); System.out.println("Age: " + user.getAge()); } } ``` 这是一个简单的自定义Spring IOC容器的实现示例,可以根据需要进行扩展和优化。 ### 回答3: 自定义一个简单的Spring IOC容器需要包含几个基本的组件和功能。以下是一个简单的实现示例: ```java import java.util.HashMap; import java.util.Map; public class CustomIOCContainer { private Map<String, Object> beanMap; // 存储bean的map public CustomIOCContainer() { beanMap = new HashMap<>(); } public void registerBean(String beanName, Object bean) { beanMap.put(beanName, bean); // 注册bean到map中 } public Object getBean(String beanName) { return beanMap.get(beanName); // 通过beanName获取对应的bean对象 } public static void main(String[] args) { CustomIOCContainer container = new CustomIOCContainer(); // 创建bean对象并注册到容器中 UserService userService = new UserServiceImpl(); container.registerBean("userService", userService); // 从容器中获取bean对象并使用 UserService userServiceFromContainer = (UserService) container.getBean("userService"); userServiceFromContainer.sayHello(); // 调用方法 // 结果输出:Hello, World! } } interface UserService { void sayHello(); } class UserServiceImpl implements UserService { @Override public void sayHello() { System.out.println("Hello, World!"); } } ``` 以上代码实现了一个简单的Spring IOC容器,核心思想是使用一个Map来存储注册的bean,通过bean的名称来获取对应的对象。在main方法中,我们创建了一个UserService对象并注册到容器中,然后通过容器的getBean方法获取UserService对象,并调用其sayHello方法进行输出。 这只是一个简化的实现示例,实际的Spring IOC容器功能还包括依赖注入、生命周期管理、AOP等复杂功能,不过以上代码可作为基本框架的起点。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值