原文系装载: http://blog.sina.com.cn/s/blog_6a7f00ed01011dyv.html

最近对spring IOC AOP 机制实现原理了解了下,在此做下整理,希望能给需要的朋友予以帮助。整理的资料来自互联网,文章开头是我写的测试代码例子,可以直接导入eclipse,别忘了导入dom4jjar包。

例子代码下载链接:http://download.csdn.net/source/630961

利用java的反射和动态代理实现IOC

在Java中,其反射和动态代理机制极其强大,我们可以通过其反射机制在运行时获取信息。而代理是一种基本的设计模式,它是一种为了提供额外的或不同的操作而插入到真实对象中的某个对象。而Java的动态代理在代理上更进一步,既能动态的创建代理对象,又能动态的调用代理方法。Java的反射和动态代理机制,使Java变得更加强大。

Spring框架这几年风头正劲,虽然使用者众多,但真正了解其内部实现原理的朋友却并不是很多。其实,了解它的内部实现机制和设计思想是很有必要的大家都知道,Spring框架的IOC和AOP部分功能强大,很值得我们学习。那么让我们在这两篇文章中分别详细的学习IOC和AOP的实现吧。

在本文中,主要讲述的是用Java的反射机制实现IOC。下面,让我们开始IOC之旅吧!

一.             Java反射机制概述与初探

Java的反射机制是Java语言的一个重要特性,Java具有的比较突出的动态机制就是反射(reflection)。通过它,我们可以获取如下信息:

1) 在运行时判断任意一个对象所属的类;

2) 在运行时获取类的对象;

3) 在运行时获得类所具有的成员变量和方法等。

下面让我们通过调用一个Java Reflection API的演示实例来见识一下反射机制的强大。

首先在IDE中建立名为reflection_proxy的Java工程,并建立存放源文件的目录src,并在src目录下分别建立org.amigo.
reflection目录和org.amigo.proxy目录来分别存放代理和反射的实例。我们在reflection目录下建立ReflectionTest.java文件,在该文件中编写代码来演示Java
Reflection API的使用。该类的代码如下所示:

  1. package org.amigo.reflection;
  2. import java.awt.Button;
  3. import java.lang.reflect.Method;
  4. import java.util.Hashtable;
  5. publicclass ReflectionTest {
  6. publicstaticvoid main(String[] args) throws Exception {
  7. ReflectionTest reflection = new ReflectionTest();
  8. reflection.getNameTest();
  9. System.out.println("");
  10. reflection.getMethodTest();
  11. }
  12. publicvoid getNameTest() throws Exception {
  13. System.out.println("===========begin getNameTest============");
  14. String name = "阿蜜果";
  15. Class cls = name.getClass();
  16. System.out.println("String类名: " + cls.getName());
  17. Button btn = new Button();
  18. Class btnClass = btn.getClass();
  19. System.out.println("Button类名: " + btnClass.getName());
  20. Class superBtnClass = btnClass.getSuperclass();
  21. System.out.println("Button的父类名: " + superBtnClass.getName());
  22. Class clsTest = Class.forName("java.awt.Button");
  23. System.out.println("clsTest name: " + clsTest.getName());
  24. System.out.println("===========end getNameTest============");
  25. }
  26. publicvoid getMethodTest() throws Exception {
  27. System.out.println("===========begin getMethodTest==========");
  28. Class cls = Class.forName("org.amigo.reflection.ReflectionTest");
  29. Class ptypes[] = new Class[2];
  30. ptypes[0] = Class.forName("java.lang.String");
  31. ptypes[1] = Class.forName("java.util.Hashtable");
  32. Method method = cls.getMethod("testMethod", ptypes);
  33. Object args[] = new Object[2];
  34. args[0] = "hello, my dear!";
  35. Hashtable<String, String> ht = new Hashtable<String, String>();
  36. ht.put("name", "阿蜜果");
  37. args[1] = ht;
  38. String returnStr = (String) method.invoke(new ReflectionTest(), args);
  39. System.out.println("returnStr= " + returnStr);
  40. System.out.println("===========end getMethodTest==========");
  41. }
  42. public String testMethod(String str, Hashtable ht) throws Exception {
  43. String returnStr = "返回值";
  44. System.out.println("测试testMethod()方法调用");
  45. System.out.println("str= " + str);
  46. System.out.println("名字= " + (String) ht.get("name"));
  47. System.out.println("结束testMethod()方法调用");
  48. return returnStr;
  49. }
  50. }

运行该例,可在控制台看到如下内容:

===========begin getNameTest============

String类名:
java.lang.String

Button类名:
java.awt.Button

Button的父类名:
java.awt.Component

clsTest name: java.awt.Button

===========end getNameTest============

===========begin getMethodTest==========

测试testMethod()方法调用

str= hello, my dear!

名字= 阿蜜果

结束testMethod()方法调用

returnStr= 返回值

===========end getMethodTest==========

分析运行结果,我们可以发现,Java的反射机制使得我们在运行时能够判断一个对象所属的类,获取对象的方法并得其进行调用,并获取方法的返回结果等功能。

二.             IOC使用的背景

在我们日常的设计中,类与类之间存在着千丝万缕的关系,如果两个类存在着强耦合关系,那么在维护时,一个类的修改很可能会牵动另一个类的关联修改,从而使得我们的维护工作步履维艰。下面让我们来看这样的一个强耦合反面例子。

首先我们建立一个Chinese.java类,该类的sayHelloWorld(String name)方法,用中文对名为name的人问好,其内容如下:

  1. package org.amigo.reflection;
  2. publicclass Chinese {
  3. publicvoid sayHelloWorld(String name) {
  4. String helloWorld = "你好," + name;
  5. System.out.println(helloWorld);
  6. }
  7. }

下面我们接着建立一个American.java类,该类的sayHelloWorld(String
name)方法,用英文对名为name的人问好,其内容如下:

  1. package org.amigo.reflection;
  2. publicclass American {
  3. publicvoid sayHelloWorld(String name) {
  4. String helloWorld = "Hello," + name;
  5. System.out.println(helloWorld);
  6. }
  7. }

最后我们编写一个测试类对这两个类的sayHelloWorld(String
name)方法进行测试,下面是该类的内容:

  1. package org.amigo.reflection;
  2. publicclass HelloWorldTest {
  3. publicstaticvoid main(String[] args) {
  4. Chinese chinese = new Chinese();
  5. chinese.sayHelloWorld("阿蜜果");
  6. American american = new American();
  7. american.sayHelloWorld("Amigo");
  8. }
  9. }

观察HelloWorldTest我们可以很清楚的看到,该类与Chinese.java类和American.java类都存在强耦合关系。

上面的例子让我们想到的是在N年以前,当我们需要某个东西时,我们一般是自己制造。但是当发展到了一定的阶段后,工厂出现了,我们可以工厂中购买我们需要的东西,这极大的方便了我们。在上例中,我们都是通过new来创建新的对象,在开发中,这种强耦合关系是我们所不提倡的,那么我们应该如何来实现这个例子的解耦呢?我们接着想到了使用工厂模式,我们需要新建一个工厂类来完成对象的创建,并采用依赖接口的方式,此时需要对代码进行如下修改:

首先建立接口类Human.java,其内容如下:

  1. package org.amigo.reflection;
  2. public interface Human {
  3. public void sayHelloWorld(String name);
  4. }

并将American.java类和Chinese.java类改为实现该接口,即类头分别改成:public
class American implements Human和public class Chinese implements Human。

接着编写HumanFactory.java工厂类,其内容为:

  1. package org.amigo.reflection;
  2. public class HumanFactory {
  3. public Human getHuman(String type) {
  4. if ("chinese".equals(type)) {
  5. return new Chinese();
  6. } else {
  7. return new American();
  8. }
  9. }
  10. }

最后我们还需要修改测试类HelloWorld.java类,修改后的内容如下:

  1. package org.amigo.reflection;
  2. public class HelloWorldTest {
  3. public static void main(String[] args) {
  4. HumanFactory factory = new HumanFactory();
  5. Human human1 = factory.getHuman("chinese");
  6. human1.sayHelloWorld("阿蜜果");
  7. Human human2 = factory.getHuman("american");
  8. human2.sayHelloWorld("Amigo");
  9. }
  10. }

观察此例我们可以看到,该类不再与具体的实现类Chinese和American存在耦合关系,而只是与它们的接口类Human存在耦合关系,具体对象的获取只是通过传入字符串来获取,很大程度上降低了类与类之间的耦合性。

但是我们还是不太满足,因为还需要通过chinese和american在类中获取实例,那么当我们需要修改时实现时,我们还需要在类中修改这些字符串,那么还有没有更好的办法呢?让我们在下节中进行继续探讨。

三.             IOC粉墨登场

IOC(Inverse of Control)可翻译为“控制反转”,但大多数人都习惯将它称为“依赖注入”。在Spring中,通过IOC可以将实现类、参数信息等配置在其对应的配置文件中,那么当需要更改实现类或参数信息时,只需要修改配置文件即可,这种方法在上例的基础上更进一步的降低了类与类之间的耦合。我们还可以对某对象所需要的其它对象进行注入,这种注入都是在配置文件中做的,Spring的IOC的实现原理利用的就是Java的反射机制, Spring还充当了工厂的角色,我们不需要自己建立工厂类。Spring的工厂类会帮我们完成配置文件的读取、利用反射机制注入对象等工作,我们可以通过bean的名称获取对应的对象。

下面让我们看看如下的模拟Spring的bean工厂类:

  1. package org.amigo.reflection;
  2. import java.io.InputStream;
  3. import java.lang.reflect.Method;
  4. import java.util.HashMap;
  5. import java.util.Iterator;
  6. import java.util.Map;
  7. import org.dom4j.Attribute;
  8. import org.dom4j.Document;
  9. import org.dom4j.Element;
  10. import org.dom4j.io.SAXReader;
  11. public class BeanFactory {
  12. private Map<String, Object> beanMap = new HashMap<String, Object>();
  13. public void init(String xml) {
  14. try {
  15. //读取指定的配置文件
  16. SAXReader reader = new SAXReader();
  17. ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
  18. //从class目录下获取指定的xml文件
  19. InputStream ins = classLoader.getResourceAsStream(xml);
  20. Document doc = reader.read(ins);
  21. Element root = doc.getRootElement();
  22. Element foo;
  23. //遍历bean
  24. for (Iterator i = root.elementIterator("bean"); i.hasNext();) {
  25. foo = (Element) i.next();
  26. //获取bean的属性id和class
  27. Attribute id = foo.attribute("id");
  28. Attribute cls = foo.attribute("class");
  29. //利用Java反射机制,通过class的名称获取Class对象
  30. Class bean = Class.forName(cls.getText());
  31. //获取对应class的信息
  32. java.beans.BeanInfo info = java.beans.Introspector.getBeanInfo(bean);
  33. //获取其属性描述
  34. java.beans.PropertyDescriptor pd[] = info.getPropertyDescriptors();
  35. //设置值的方法
  36. Method mSet = null;
  37. //创建一个对象
  38. Object obj = bean.newInstance();
  39. //遍历该bean的property属性
  40. for (Iterator ite = foo.elementIterator("property"); ite.hasNext();) {
  41. Element foo2 = (Element) ite.next();
  42. //获取该property的name属性
  43. Attribute name = foo2.attribute("name");
  44. String value = null;
  45. //获取该property的子元素value的值
  46. for(Iterator ite1 = foo2.elementIterator("value"); ite1.hasNext();) {
  47. Element node = (Element) ite1.next();                                          value = node.getText();
  48. break;
  49. }
  50. for (int k = 0; k < pd.length; k++) {
  51. if (pd[k].getName().equalsIgnoreCase(name.getText())) {                                             mSet = pd[k].getWriteMethod();
  52. //利用Java的反射极致调用对象的某个set方法,并将值设置进去                                              mSet.invoke(obj, value);
  53. }
  54. }
  55. }
  56. //将对象放入beanMap中,其中key为id值,value为对象
  57. beanMap.put(id.getText(), obj);
  58. }
  59. } catch (Exception e) {
  60. System.out.println(e.toString());
  61. }
  62. }
  63. public Object getBean(String beanName) {
  64. Object obj = beanMap.get(beanName);
  65. return obj;
  66. }
  67. public static void main(String[] args) {
  68. BeanFactory factory = new BeanFactory();
  69. factory.init("config.xml");
  70. JavaBean javaBean = (JavaBean) factory.getBean("javaBean");
  71. System.out.println("userName=" + javaBean.getUserName());
  72. System.out.println("password=" + javaBean.getPassword());
  73. }
  74. }

该类的init(xml)方法,通过指定的xml来给对象注入属性,为了对该类进行测试,我还需要新建一个JavaBean和在src目录下新建一个名为config.xml的配置文件。JavaBean的内容如下:

  1. package org.amigo.reflection;
  1. public class JavaBean {
  2. private String userName;
  3. private String password;
  4. public String getPassword() {
  5. return password;
  6. }
  7. public String getUserName() {
  8. return userName;
  9. }
  10. public void setUserName(String userName) {
  11. this.userName = userName;
  12. }
  13. public void setPassword(String password) {
  14. this.password = password;
  15. }
  16. }

这个简单bean对象中有两个属性,分别为userName和password,下面我们在配置文件config.xml中对其属性注入对应的属性值。配置文件内容如下:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans>
  3. <bean id="javaBean" class="org.amigo.reflection.JavaBean">
  4. <property name="userName">
  5. <value>阿蜜果</value>
  6. </property>
  7. <property name="password">
  8. <value>12345678</value>
  9. </property>
  10. </bean>
  11. </beans>

类与配置文件都完成后,可以运行BeanFactory.java文件,控制台显示内容为:

userName=阿蜜果

password=12345678

可以看到,虽然在main()方法中没有对属性赋值,但属性值已经被注入,在BeanFactory类中的Class
bean = Class.forName(cls.getText());通过类名来获取对应的类,mSet.invoke(obj, value);通过invoke方法来调用特定对象的特定方法,实现的原理都是基于Java的反射机制,在此我们有一次见证了Java反射机制的强大。

当然,这只是对IOC的一个简单演示,在Spring中,情况要复杂得多,例如,可以一个bean引用另一个bean,还可以有多个配置文件、通过多种方式载入配置文件等等。不过原理还是采用Java的反射机制来实现IOC的。

四.             总结

在本文中,笔者通过讲述Java反射机制概述与初探、IOC使用的背景、IOC粉墨登场等内容,演示了Java反射机制API的强大功能,并通过编写自己的简单的IOC框架,让读者更好的理解了IOC的实现原理。

本文通过IOC的一个简要实现实例,模拟了Spring中IOC的实现,虽然只是完成了Spring中依赖注入的一小部分工作,但是很好的展现了Java反射机制在Spring中的应用,能使我们能更好的从原理上了解IOC的实现,也能为我们实现自己的准Spring框架提供方案,有兴趣的朋友可以通过Spring的源码进行IOC的进一步的学习。

spring IOC 实现原理模拟实现的更多相关文章

  1. spring ioc aop 原理

    spring ioc aop 原理 spring ioc aop 的原理 spring的IoC容器是spring的核心,spring AOP是spring框架的重要组成部分. 在传统的程序设计中,当调 ...

  2. Spring IoC底层原理

    -------------------siwuxie095                                 Spring IoC 底层原理         1.IoC 即 Invers ...

  3. Spring IOC设计原理解析:本文乃学习整理参考而来

    Spring IOC设计原理解析:本文乃学习整理参考而来 一. 什么是Ioc/DI? 二. Spring IOC体系结构 (1) BeanFactory (2) BeanDefinition 三. I ...

  4. Spring框架系列(6) - Spring IOC实现原理详解之IOC体系结构设计

    在对IoC有了初步的认知后,我们开始对IOC的实现原理进行深入理解.本文将帮助你站在设计者的角度去看IOC最顶层的结构设计.@pdai Spring框架系列(6) - Spring IOC实现原理详解 ...

  5. Spring框架系列(7) - Spring IOC实现原理详解之IOC初始化流程

    上文,我们看了IOC设计要点和设计结构:紧接着这篇,我们可以看下源码的实现了:Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的. ...

  6. Spring框架系列(8) - Spring IOC实现原理详解之Bean实例化(生命周期,循环依赖等)

    上文,我们看了IOC设计要点和设计结构:以及Spring如何实现将资源配置(以xml配置为例)通过加载,解析,生成BeanDefination并注册到IoC容器中的:容器中存放的是Bean的定义即Be ...

  7. 深入理解Spring IOC工作原理

    为什么会出现spring,spring出现解决了什么问题? 1.分析普通多层架构存在的问题 JSP->Servlet->Service->Dao 层与层之间的依赖很强,属于耦合而且是 ...

  8. Spring IOC Container原理解析

    Spring Framework 之 IOC IOC.DI基础概念 关于IOC和DI大家都不陌生,我们直接上martin fowler的原文,里面已经有DI的例子和spring的使用示例 <In ...

  9. [Spring框架]Spring IOC的原理及详解。

    这里感谢 CSDN 的原博客:http://blog.csdn.net/m13666368773/article/details/7802126 看后  受益匪浅,这里再重温一遍Spring IOC ...

  10. java框架篇---spring IOC 实现原理

    IOC(DI):其实这个Spring架构核心的概念没有这么复杂,更不像有些书上描述的那样晦涩.java程序员都知道:java程序中的每个业务逻辑至少需要两个或以上的对象来协作完成,通常,每个对象在使用 ...

随机推荐

  1. IDEA 2023.2 最新安装使用教程(附激活码,亲测好用)

    申明:本教程 IDEA 补丁.补丁均收集于网络,请勿商用,仅供个人学习使用,如有侵权,请联系作者删除.若条件允许,希望大家购买正版 ! idea激活码使用教程 Step1 第一步下载IDEA软件 ID ...

  2. ECharts实现雷达图详解

    ECharts 是一款由百度开源的数据可视化工具,它提供了丰富的图表类型,如折线图.柱状图.饼图.散点图.雷达图.地图.K线图.热力图.仪表盘等,以及丰富的交互功能.ECharts 组件的核心功能实现 ...

  3. 【YashanDB知识库】汇聚库23.1环境发生coredump

    [标题]汇聚库23.1环境发生coredump [问题分类]数据库错误 [关键词]YashanDB, 汇聚库, coredump [问题描述]在23.1.1.200版本数据库环境创建dblink.视图 ...

  4. div中多行内容垂直居中显示

    div中多行内容垂直居中显示 一.css 样式 .wrap { height: 200px; width: 200px; border: 1px solid #232323; display: fle ...

  5. docker启动问题: Job for docker.service failed because the control process exited with error code. See "systemctl status docker.service" and "journalctl -xe" for details.

    系统环境:centos 7 docker版本:Docker version 26.1.4, build 5650f9b 问题:Job for docker.service failed because ...

  6. Angular 18+ 高级教程 – Component 组件 の @let Template Local Variables

    前言 Angular 在 v18.1 推出了 Template 新语法 @let. 这个 @let 和上一篇教的 Control Flow @if, @for, @swtich, @defer 语法上 ...

  7. Angular 18+ 高级教程 – Angular Configuration (angular.json)

    前言 记入一些基本的配置. Setup IP Address.SSL.Self-signed Certificate 如果你对 IP Address.SSL.Self-signed Certifica ...

  8. Docker安装(安装Docker-CE)(三)

    现版本安装Docker已经非常简单了,有很多种方式,而自17年开始,Docker分为Docker-CE(社区版).Docker-EE(企业版),另外Docker-IO是较早的版本,通常用的都是Dock ...

  9. 《Vue.js 设计与实现》读书笔记(1-3章)

    第 1 章.权衡的艺术 命令式 or 声明式 命令式:关注过程 声明式:关注结果 声明式直接声明想要的结果,框架帮用户封装好命令式的代码,所以在封装的过程中要做一些其他的事情来(生成要做的事情/找出差 ...

  10. 左值 <->右值

    左值引用指向左值 右值引用指向右值 int a = 5; int &ref_a = a; // 左值引用指向左值,编译通过 int &ref_a = 5; // 左值引用指向了右值,会 ...