spring如何实现注入
IOC(Inverse of Control)可翻译为“控制反转”,但大多数人都习惯将它称为“依赖注入”。在Spring中,通过IOC可以将实现类 、参数信息等配置在其对应的配置文件中 ,那么当需要更改实现类或参数信息时,只需要修改配置文件即可,这种方法在上例的基础上更进一步的降低了类与类之间的耦合。我们还可以对某对象所需要的其它对象进行注入 ,这种注入都是在配置文件中做的,Spring的IOC的实现原理利用的就是Java的反射机制,Spring还充当了工厂的角色,我们不需要自己建立工厂类 。Spring的工厂类会帮我们完成配置文件的读取、利用反射机制注入对象等工作,我们可以通过bean的名称获取对应的对象。
定义一个bean src/main/java/spring_IOC/JavaBean.java
package spring_IOC;
/**
* Created by tom on 2016/5/18.
*/
public class JavaBean {
String username;
String password;
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
}
实例化一个类,利用反射,将其注入值 src/main/java/spring_IOC/SetValueByReflection.java
package spring_IOC;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
/**
* Created by tom on 2016/5/18.
*/
public class SetValueByReflection {
/**
* 实例化一个类,利用反射,将其注入值
*/
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {
JavaBean bean = new JavaBean();
//获取指定类的指定方法,
Class c = Class.forName("spring_IOC.JavaBean");
Method method = c.getMethod("setUsername", new Class[]{String.class});
//对带有指定参数的指定对象调用由此 Method 对象表示的底层方法,调用对象javaBean的setuserName方法,参数为"hello world"
method.invoke(bean, "hello world");
System.out.println(bean.getUsername());
}
}
实例化一个类,利用反射,用个map来摸拟在xml获取的属性名及值 src/main/java/spring_IOC/SetValueByReflectionLoop.java
package spring_IOC;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
/**
* Created by tom on 2016/5/18.
*/
public class SetValueByReflectionLoop {
static Logger log = LoggerFactory.getLogger(BeanFactory.class);
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException, IntrospectionException {
//这个map对象用来模拟对属性文件解析获了的属性名与值
Map<String, Object> map = new HashMap<>();
map.put("username", "tomLuo");
map.put("password", "954");
Class bean = Class.forName("spring_IOC.JavaBean");
Object obj = bean.newInstance();
//获取对应class信息
BeanInfo info = Introspector.getBeanInfo(bean);
// 遍历指定类的属性
PropertyDescriptor[] propertys = info.getPropertyDescriptors();
for (int j = 0; j < propertys.length; j++) {
System.out.println("属性:" + propertys[j].getName());
}
//获取其属性描述
java.beans.PropertyDescriptor pd[] = info.getPropertyDescriptors();
Method mSet = null;
for (int k = 0; k < pd.length; k++) {
if (map.containsKey(pd[k].getName())) {
Object value=map.get(pd[k].getName());//将对应的属性值取出来
mSet = pd[k].getWriteMethod();
log.info("{} {} {}", pd[k].getName(), pd[k].getWriteMethod().getName(), pd[k].getReadMethod().getName());//password setPassword getPassword
mSet.invoke(obj, value);//利用反射将567注入到bean 这儿实验将每个set方法的值都设置为567
}
}
//将对象放入beanMap中,其中key为id值,value为对象
JavaBean javaBean1 = (JavaBean) obj;
System.out.println("userName=" + javaBean1.getUsername());
System.out.println("password=" + javaBean1.getPassword());
}
}
加载xml文件,利用反射,将其注入值 src/main/resources/config.xml
<?xml version="1.0" encoding="utf-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans">
<bean id="javaBean" class="spring_IOC.JavaBean">
<property name="username" value="tom"/>
<property name="password" value="123"/>
</bean>
</beans>
Bean工厂主要用来解析xml文件,获取属性名及值,然后利用反射,将其注入值 src/main/java/spring_IOC/BeanFactory.java
package spring_IOC;
import org.dom4j.Attribute;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.InputStream;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
/**
* Created by tom on 2016/5/18.
*/
public class BeanFactory {
static Logger log = LoggerFactory.getLogger(BeanFactory.class);
private Map<String, Object> beanMap = new HashMap<String, Object>();
/**
* bean工厂的初始化
*
* @param xml
*/
public void init(String xml) {
try {
//读取指定的配置文件
SAXReader reader = new SAXReader();
//从class目录下获取指定的配置文件
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
InputStream inputStream = classLoader.getResourceAsStream(xml);
//读取xml文件
Document document = reader.read(inputStream);
//获取跟节点
Element root = document.getRootElement();
//遍历bean节点
Element foo;
for (Iterator iteBean = root.elementIterator("bean"); iteBean.hasNext(); ) {
foo = (Element) iteBean.next();
//获取bean的属性id和class
Attribute id = foo.attribute("id");
Attribute cls = foo.attribute("class");
//利用java反射机制,通过class的名称获取Class对象
log.debug("{}", cls.getText());
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;
//创建一个对象(创建此 Class 对象所表示的类的一个新实例。如同用一个带有一个空参数列表的 new 表达式实例化该类。如果该类尚未初始化,则初始化这个类。)
Object obj = bean.newInstance();
//遍历该bean的property属性
for (Iterator iteProperty = foo.elementIterator("property"); iteProperty.hasNext(); ) {
Element elementProperty = (Element) iteProperty.next();
//获取该property的name属性
Attribute name = elementProperty.attribute("name");
//读取该属性值
String value = elementProperty.attribute("value").getText();
//String value = null;
//获取该property的子元素value的值
//for(Iterator iteValue = elementProperty.elementIterator("value");iteValue.hasNext();){
// Element elementValue = (Element)iteValue.next();
// value = elementValue.getText();
// break;
//}
for (int k = 0; k < pd.length; k++) {
log.info(pd[k].getName());
if (pd[k].getName().equalsIgnoreCase(name.getText())) {
mSet = pd[k].getWriteMethod();//
log.info(mSet.getName());
//利用Java的反射机制调用对象的某个set方法,并将值设置进去
mSet.invoke(obj, value);//通过invoke方法来调用特定对象的特定方法,实现的原理都是基于Java的反射机制
}
}
}
//将对象放入beanMap中,其中key为id值,value为对象
beanMap.put(id.getText(), obj);
}
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 通过bean的id获取bean的对象.
*
* @param beanName bean的id
* @return 返回对应对象
*/
public Object getBean(String beanName) {
Object obj = beanMap.get(beanName);
return obj;
}
public static void main(String[] args) {
BeanFactory factory = new BeanFactory();
factory.init("config.xml");
JavaBean javaBean = (JavaBean) factory.getBean("javaBean");
System.out.println("userName=" + javaBean.getUsername());
System.out.println("password=" + javaBean.getPassword());
}
}
可以看到,虽然在main()方法中没有对属性赋值,但属性值已经被注入,在BeanFactory类中的Class bean = Class.forName(cls.getText()); 通过类名来获取对应的类,mSet.invoke(obj, value);通过invoke方法来调用特定对象的特定方法,实现的原理都是基于Java的反射机制
源码浏览: https://github.com/tomlxq/job-test/tree/master/java-base/src/main/java/spring_IOC