Spring有两大核心特性IOC和AOP,今天我们来聊的是IOC。相信很多同学每天都在使用Spring的IOC特性,但对其原理及实现理解比较模糊,那么今天就借助Spring容器实现原理,简单说说IOC。说到IOC就涉及到两个概念控制反转(InversionofControl)、依赖注入(DependencyInjection)。 首先来看什么是控制反转 在介绍控制反转之前呢我们看一段最早的代码PublicclassPersonServiceBean{PrivatePersonDaoBeanpersonDaonewPersonDaoBean();Publicvoidsave(Personperson){personDao。insert(person);}} PersonDaoBean是在应用内部创建及维护的。所谓控制反转就是应用本身不负责依赖对象的创建及维护,依赖对象的创建及维护是由外部容器负责的。这样控制反转权就由应用转到了外部容器,控制权的转移就是所谓反转。 用Spring的写法:把依赖对象交给外部容器去负责PublicclassPersonServiceBean{PrivatePersonDaoBeanpersonD把容器创建好的对象依赖注入进PersonServiceBean,通过构造方法或set方法PublicPersonServiceBean(PersonDaoBeanpersonDao){this。personDaopersonD}或PublicvoidsetPersonDao(PersonDaoBeanpersonDao){this。personDaopersonD}Publicvoidsave(Personperson){personDao。insert(person);}} 其次来看什么是依赖注入 所谓依赖注入:就是在运行期间,由外部容器动态的将依赖对象注入到组件中。 简单的说,Spring就是通过工厂反射将我们的bean放到它的容器中,当我们想用某个bean的时候,只需要调用getBean(beanId)方法。 依赖注入的方式: 第一。构造器注入; 第二。set方法注入; 第三。使用Field注入(用于注解方式) 然后我们写一段简单的模拟Spring的代码来实现IOC容器 思路:Spring容器的原理实现主要依赖于反射。过程其实就是通过解析xml文件,获取到用户配置的bean,然后通过反射将这些bean进行存储(放到集合中),然后对我提供一个getBean方法,以便我们获取到这些bean。下面是一段简单的模拟代码:packagecom。spring。importjava。util。HashMimportjava。util。Limportjava。util。Mimportorg。jdom。Dimportorg。jdom。Eimportorg。jdom。input。SAXBimportorg。jdom。xpath。XPpublicclassClassPathXmlApplicationContextimplementsBeanFactory{容器的核心,用来存放注入的BeanprivateMapString,ObjectcontainernewHashMapString,Object();解析xml文件,通过反射将配置的bean放到container中publicClassPathXmlApplicationContext(StringfileName)throwsException{SAXBuildersbnewSAXBuilder();Documentdocsb。build(this。getClass()。getClassLoader()。getResourceAsStream(fileName));Elementrootdoc。getRootElement();ListlistXPath。selectNodes(root,beansbean);扫描配置文件中的beanfor(inti0;ilist。size();i){Elementbean(Element)list。get(i);Stringidbean。getAttributeValue(id);Stringclazzbean。getAttributeValue(class);反射ObjectoClass。forName(clazz)。newInstance();container。put(id,o);}}OverridepublicObjectgetBean(Stringid){returncontainer。get(id);}} 说明:首先声明一个存放bean的Map,然后通过jdom解析配置文件,循环遍历所有的节点,并通过反射将它们放到我们之前声明的Map中。然后提供一个getBean的方法,让我们可以通过beanId来获取到我们想要的bean。 下面是一个简单的xml配置文件:?xmlversion1。0encodingUTF8?beansbeanidEclasscom。spring。factory。EnglandbeanidSclasscom。spring。factory。SpainbeanidPclasscom。spring。factory。Portugalbeans 客户端通过调用前面的ClassPathXmlApplicationContext,来加载上面的配置文件,然后就可以通过Id来获得我们需要的bean了:packagecom。spring。publicclassTest{publicstaticvoidmain(String〔〕args)throwsException{加载配置文件BeanFactoryfnewClassPathXmlApplicationContext(applicationContext。xml);英格兰Objectoef。getBean(E);Teame(Team)e。say();西班牙Objectosf。getBean(S);Teams(Team)s。say();葡萄牙Objectopf。getBean(P);Teamp(Team)p。say();}} 输出结果:England:我们是欧洲的中国队,不在乎这次小组没出线。。。Spain:我们是两届欧洲杯冠军、一届世界杯冠军!Portugal:我们的C罗一个顶十个! 其它代码:工厂接口packagecom。spring。publicinterfaceBeanFactory{ObjectgetBean(Stringid);}Team接口packagecom。spring。publicinterfaceTeam{voidsay();}英格兰packagecom。spring。publicclassEnglandimplementsTeam{publicvoidsay(){System。out。println(England:我们是欧洲的中国队,不在乎这次小组没出线。。。);}}西班牙packagecom。spring。publicclassSpainimplementsTeam{Overridepublicvoidsay(){System。out。println(Spain:我们是两届欧洲杯冠军、一届世界杯冠军!);}}葡萄牙packagecom。spring。publicclassPortugalimplementsTeam{Overridepublicvoidsay(){System。out。println(Portugal:我们的C罗一个顶十个!);}} 以上内容是对Spring的一个简单模拟,当然Spring远比这个要复杂的多,也强大的多,而且获取bean的方式也不止通过工厂这一种。这里只是做一个粗略的Demo说说对容器的简单理解,向Spring致敬。 扩展,那么如果有property属性的bean该如何初始化呢?(依赖注入) 如xml配置方式:bean有property属性,包含name和ref。 这里提供思路,感兴趣的同学可以自行实现,也可参看spring的源码。 思路: 1。获取Bean的set方法setA反射 2。获取Bean的声明的属性。equals(propertyDefine。getName()) 3。获取Bean的ref通过ref作为Key从Map中获取到实例对象 4。setA。invoke(Bean,实例对象)反射 整个原理大致分三步: 1。读取xml文件 2。实例化bean保存到map中 3。为有property的Bean注入Bean0对象 说了这么多,那么IOC有什么好处呢 降低组件的耦合,使各层解耦。bean对象交于容器管理。 小结 相信读完此篇会对SpringIOC的概念和实现原理更加清晰。 控制反转:就是应用本身不负责依赖对象的创建及维护,依赖对象的创建及维护是由外部容器Spring负责。 依赖注入:由外部容器Spring动态的将依赖对象注入到组件中。 实现原理:JAVA反射机制。