`
天府小茶客
  • 浏览: 12681 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

java动态代理实现原理

    博客分类:
  • java
阅读更多

java动态代理与反射

 

引言:

java的代理分为静态代理与动态代理,静态代理有其固有缺点,即代码量大,缺乏灵活性。故我们主要关注的还是动态代理。而动态代理是基于java反射机制实现的一种代理模式。

动态代理的用处:动态代理的用处很广。比如,我们常用的springAOP,即是通过动态代理技术来实现的。

 

 

要理解动态代理,首先需要了解java反射的基本概念及API方法。在这里假设你已经知道java反射的基础知识,在这个前提下,我们通过一个例子程序,即可说明动态代理的实现原理:

 

interface Manager {
	public void modify1();
	public void modify2();
}

class ManagerImpl implements Manager {
	public void modify1() {
		System.out.println("*******ManagerImpl 的 modify()1方法被调用*******");
	}
	public void modify2() {
		System.out.println("*******ManagerImpl 的 modify()2方法被调用*******");
	}
}

class ManagerImpl2 implements Manager {
	public void modify1() {
		System.out.println("*******ManagerImpl2 的 modify()1方法被调用*******");
	}
	public void modify2() {
		System.out.println("*******ManagerImpl2 的 modify()2方法被调用*******");
	}
}

class BusinessHandler implements InvocationHandler {
	private Object object = null;

	public BusinessHandler(Object object) {
		this.object = object;
	}

	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
			throws Throwable {
		System.out.println("do something before method");
		Object ret = method.invoke(this.object, args);
		System.out.println("do something after method");
		return ret;
	}
}

public class Client {
	public static void main(String[] args) {
		// 元对象(被代理对象)
		ManagerImpl managerImpl = new ManagerImpl();
		ManagerImpl2 managerImpl2 = new ManagerImpl2();
		// 业务代理类
		BusinessHandler securityHandler = new BusinessHandler(managerImpl);
		// 获得代理类($Proxy0 extends Proxy implements Manager)的实例.
		Manager managerProxy = (Manager) Proxy.newProxyInstance(managerImpl
				.getClass().getClassLoader(), managerImpl.getClass()
				.getInterfaces(), securityHandler);
		
		System.out.println("——————");
                //通过反射,可以知晓,在内存中实际生成了一个$Proxy0的类,亦可通过反射看出这个类的代码组成
		//这里可以通过运行结果证明managerProxy是Proxy的一个实例,这个实例实现了Manager接口  
        System.out.println(managerProxy instanceof Proxy);  
          
        //这里可以看出managerProxy的Class类是$Proxy0,这个$Proxy0类继承了Proxy,实现了Manager接口  
        System.out.println("managerProxy的Class类是:"+managerProxy.getClass().toString());  
          
        System.out.print("managerProxy中的属性有:");  
          
        Field[] field=managerProxy.getClass().getDeclaredFields();  
        for(Field f:field){  
            System.out.print(f.getName()+", ");  
        }  
          
        System.out.print("\n"+"managerProxy中的方法有:");  
        
        Method[] method=managerProxy.getClass().getDeclaredMethods();  
          
        for(Method m:method){  
            System.out.print(m.getName()+", ");  
        }  
          
        System.out.println("\n"+"managerProxy的父类是:"+managerProxy.getClass().getSuperclass());  
          
        System.out.print("\n"+"managerProxy实现的接口是:");  
          
        Class<?>[] interfaces=managerProxy.getClass().getInterfaces();  
          
        for(Class<?> i:interfaces){  
            System.out.print(i.getName()+", ");  
        }  
        
        System.out.println();
        System.out.println("——————");
		
		managerProxy.modify1();
		managerProxy.modify2();
		System.out.println("——————");
		// 业务代理类,用以演示动态代理的优势:不用增加格外代码,只需增加另一个代理处理类,然后绑定到一个新的InvocationHandler实例上,即可实现扩充
		securityHandler = new BusinessHandler(managerImpl2);
		// 获得代理类($Proxy0 extends Proxy implements Manager)的实例.
		managerProxy = (Manager) Proxy.newProxyInstance(managerImpl2
						.getClass().getClassLoader(), managerImpl2.getClass()
						.getInterfaces(), securityHandler);
		managerProxy.modify1();
		managerProxy.modify2();
	}
}

 执行结果:

 

 

——————
true
managerProxy的Class类是:class testReflect_10.$Proxy0
managerProxy中的属性有:m4, m1, m0, m3, m2, 
managerProxy中的方法有:equals, toString, hashCode, modify1, modify2, 
managerProxy的父类是:class java.lang.reflect.Proxy

managerProxy实现的接口是:testReflect_10.Manager, 
——————
do something before method
*******ManagerImpl 的 modify()1方法被调用*******
do something after method
do something before method
*******ManagerImpl 的 modify()2方法被调用*******
do something after method
——————
do something before method
*******ManagerImpl2 的 modify()1方法被调用*******
do something after method
do something before method
*******ManagerImpl2 的 modify()2方法被调用*******
do something after method

 说明:

 

1InvocationHandler是个代理处理类的父类(接口),只定义了一个invoke方法。BusinessHandler继承了该接口,并实现了invoke方法;

2securityHandler = new BusinessHandler(managerImpl):实例化一个代理处理类,并传入一个要代理的真实“对象”:managerImpl

3(Manager) Proxy.newProxyInstance(managerImpl 

                .getClass().getClassLoader(), managerImpl.getClass() 

 

                .getInterfaces(), securityHandler) 方法:创建了一个$Proxy0类的实例(不要问我是怎么知道的,代码里已经很清楚的看到,是可以通过反射打印出来的),这个实例继承了Proxy类,也实现了要代理的那个Manager接口,同时,还传入一个BusinessHandler的实例

 

4、重点来了,managerProxy.modify1(),这句代码执行的时候,为什么会调用BusinessHandler invoke()方法?答案:也是通过反射,让我看到了那个$Proxy0类的代码(你可以通过反射的各种方法去将这个类的所有信息都可以打印出来)。其实网上也有很多同学已经帮我们去辛辛苦苦扣下来了$Proxy0类的代码,如是:

 

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;

public final class $Proxy0 extends Proxy implements Manager{
private static Method m4;
private static Method m1;
private static Method m0;
private static Method m3;
private static Method m2;

static {
   try {
    m4 =Class.forName("java.lang.Object").getMethod("equals",
     new Class[] { Class.forName("java.lang.Object") });

    m1 =Class.forName("java.lang.Object").getMethod("toString",
     new Class[0]);

m0 =Class.forName("java.lang.Object").getMethod("hashCode",
      new Class[0]);

    m3 =Class.forName("com.ml.test.Manager").getMethod("modify1",
     new Class[0]);

    m2 =Class.forName("com.ml.test.Manager").getMethod("modify2",
     new Class[0]);

   } catch (NoSuchMethodExceptionnosuchmethodexception) {
    throw newNoSuchMethodError(nosuchmethodexception.getMessage());
   } catch(ClassNotFoundException classnotfoundexception) {
    throw newNoClassDefFoundError(classnotfoundexception.getMessage());
   }
} //static

public $Proxy0(InvocationHandler invocationhandler) {
  super(invocationhandler);
}

@Override
public final boolean equals(Object obj) {
   try {
           return ((Boolean) super.h.invoke(this, m4, new Object[] { obj })).booleanValue();
      } catch (Throwable throwable) {
           throw newUndeclaredThrowableException(throwable);
      }
}

@Override
public final int hashCode() {
   try {
           return ((Integer)super.h.invoke(this, m0, null)).intValue();
      } catch (Throwable throwable) {
           throw newUndeclaredThrowableException(throwable);
      }
}

public final void modify1() {
   try {
          super.h.invoke(this, m3, null);
          return;
       } catch (Error e) {
       } catch (Throwable throwable) {
          throw new UndeclaredThrowableException(throwable);
       }
}

public final void modify2() {
   try {
          super.h.invoke(this, m2, null);
          return;
       } catch (Error e) {
       } catch (Throwable throwable) {
          throw new UndeclaredThrowableException(throwable);
       }
}

@Override
public final String toString() {
   try {
          return (String) super.h.invoke(this, m1,null);
      } catch (Throwable throwable) {
          throw new UndeclaredThrowableException(throwable);
      }
 }
}

 看到关键代码没?这个:

 

 

public final void modify1() {
   try {
          super.h.invoke(this, m3, null);
          return;

 明白了吧?代理类每次执行方法时,实际就是通过BusinessHandler这个真正的代理处理类的invoke方法来执行的。invoke方法的:method.invoke(this.object, args)这个方法,是java的反射技术,通过这个方法其实真正执行了managerImplmodify1()这些方法。

 

 

——————这下明白了吧,动态代理的实现步骤:

1、继承并实现InvocationHandler接口及其invoke方法,这个方法体的实现其实也比较固定,就如例子程序那样;

2、通过ProxynewProxyInstance方法创建一个代理实例,这个创建方法也会把真正处理方法执行的InvocationHandler绑定在一块。

3、再用第二步的proxy实例执行方法时,实际是调用了InvocationHandlerinvoke方法,最终执行了真实的managerImplmodify1()方法。

 

 

最后,我再画一个图吧:



 

希望对大家理解动态代理有所帮助!谢谢。

 

 

  • 大小: 66.2 KB
1
0
分享到:
评论

相关推荐

    java 动态代理实现AOP

    ### Java动态代理实现AOP详解 #### 一、引言 随着软件开发复杂度的提升,传统的面向对象编程(OOP)已经难以满足现代软件工程的需求。为了更好地管理跨切面的关注点,如日志记录、性能监控、安全控制等,面向切面...

    java动态代理实现详解

    Java 动态代理是Java平台提供的一种强大的工具,它允许我们在运行时动态生成代理类,这些代理类可以实现一组指定的接口,同时还能在方法调用前后添加自定义的行为。这种机制大大简化了面向切面编程(AOP)和事件监听...

    JAVA实现动态代理的简单流程

    ### Java动态代理的实现原理 Java动态代理主要依赖于`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口。`Proxy`类提供了创建动态代理对象的方法,而`InvocationHandler`则定义了代理对象...

    Java动态代理实现数据源连接池

    总的来说,Java动态代理实现数据源连接池的核心思想是利用`InvocationHandler`来拦截对数据库操作的调用,通过代理对象在每次操作前后进行连接的获取和释放,从而实现连接池的逻辑。这种方式虽然相比现成的连接池库...

    java动态代理 经典文章(word 2007格式的)

    - `设计模式:用Java动态代理实现AOP.docx`可能包含了如何使用Java动态代理来实现设计模式,尤其是代理模式,以达到AOP的效果。 - `Spring AOP入门实例详解.docx`可能涵盖了Spring AOP的基本概念、配置以及如何...

    java动态代理实例

    首先,我们需要了解Java动态代理的基本原理。在Java中,动态代理主要通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口来实现。`Proxy`类用于创建代理对象,而`InvocationHandler`接口...

    Java动态代理(Spring Aop原理)

    动态代理提供了创建代理对象的机制,而Spring AOP则是基于动态代理实现的面向切面编程框架,简化了在多处插入相同逻辑的复杂度。在实际开发中,理解并熟练运用这两个技术对于提高代码质量和可维护性至关重要。

    用Java动态代理实现委托模式

    下面我们将详细探讨如何使用Java动态代理实现委托模式,以及相关的源码和工具应用。 首先,理解委托模式的基本概念。委托模式通常由三部分组成:委托者(Delegate)、被委托者(Subject)和客户端(Client)。委托...

    java动态代理实现数据库连接池

    ### Java动态代理实现数据库连接池 #### 背景与挑战 在开发应用程序时,数据库连接池是一个常用且重要的组成部分。频繁地建立和断开数据库连接不仅效率低下,还可能导致性能瓶颈。为了解决这一问题,引入了数据库...

    java动态代理机制分析及拓展

    Java动态代理机制的优点在于灵活性和便捷性,它允许在运行时动态创建符合特定接口的代理对象,无需对原有代码做任何修改,使得我们可以方便地实现诸如日志、事务控制等跨切面的逻辑。不过,动态代理仅限于代理接口,...

    Java代理模式Java动态代理

    无论是通过代理模式控制对象的访问,还是利用Java动态代理实现特定时刻的行为增强,都是在不改变现有代码结构的前提下实现功能扩展的有效手段。这对于提高代码的可维护性和灵活性具有重要意义。

    Java动态代理helloworld

    本篇文章将深入探讨Java动态代理的概念、原理以及如何通过一个简单的"Hello, World!"示例来理解它。 动态代理,顾名思义,是在程序运行时动态地生成代理对象。与静态代理(编译时已知)相比,动态代理更加灵活,...

    java-用Java动态代理实现AOP.pdf

    从文件内容可以看出,文档主要讨论了如何使用Java动态代理来实现AOP的基本原理。文档中提供了一个简单的例子,其中定义了一个业务接口BusinessInterface和一个实现了该接口的BusinessObject类。BusinessObject类中的...

    Java JDK动态代理实现原理实例解析

    Java JDK动态代理实现原理实例解析 Java JDK动态代理实现原理实例解析是指通过Java JDK提供的代理机制来实现动态代理的原理和实例解析。动态代理是一种非常重要的设计模式,它可以在运行时生成代理类,从而实现了...

    动态代理原理实例Demo

    动态代理主要有两种实现方式:JDK自带的`java.lang.reflect.Proxy`类和`java.lang.invoke.MethodHandle`API。这里主要讨论JDK的Proxy类,因为这是最常见且相对简单的实现方式。 1. **Proxy类的使用** - **Proxy....

    反射实现 AOP 动态代理模式(Spring AOP 的实现原理)

    在Java中,可以使用java.lang.reflect包下的相关类和接口实现动态代理。 例如,通过使用java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口,我们可以在运行时创建一个代理对象。代理对象在调用...

    Java动态代理案例演示代码

    Java动态代理是Java语言提供的一种强大的机制,它允许在运行时创建具有特定接口的代理类对象,这些代理类对象能够对方法调用进行拦截和增强。动态代理在很多场景下非常有用,例如AOP(面向切面编程)、事件监听、...

Global site tag (gtag.js) - Google Analytics