java动态代理中的invoke方法是如何被自动调用的「建议收藏」

java动态代理中的invoke方法是如何被自动调用的「建议收藏」Java中动态代理的实现,关键就是这两个东西:Proxy、InvocationHandler,下面从InvocationHandler接口中的invoke方法入手,简单说明一下Java如何实现动态代理的。        首先,invoke方法的完整形式如下: Java代码  public Object invoke(Object proxy, Method m

大家好,又见面了,我是你们的朋友全栈君。

一、动态代理与静态代理的区别。
(1)Proxy类的代码被固定下来,不会因为业务的逐渐庞大而庞大;
(2)可以实现AOP编程,这是静态代理无法实现的;
(3)解耦,如果用在web业务下,可以实现数据层和业务层的分离。
(4)动态代理的优势就是实现无侵入式的代码扩展。
   静态代理这个模式本身有个大问题,如果类方法数量越来越多的时候,代理类的代码量是十分庞大的。所以引入动态代理来解决此类问题

二、动态代理

Java中动态代理的实现,关键就是这两个东西:Proxy、InvocationHandler,下面从InvocationHandler接口中的invoke方法入手,简单说明一下Java如何实现动态代理的。 
        首先,invoke方法的完整形式如下: 

Java代码  
收藏代码

  1. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable  
  2.     {  
  3.   
  4.         method.invoke(obj, args);  
  5.   
  6.         return null;  
  7.     }  

        首先猜测一下,method是调用的方法,即需要执行的方法;args是方法的参数;proxy,这个参数是什么?以上invoke()方法的实现即是比较标准的形式,我们看到,这里并没有用到proxy参数。查看JDK文档中Proxy的说明,如下: 

Java代码  
收藏代码

  1. A method invocation on a proxy instance through one of its proxy interfaces will be dispatched to the invoke method of the instance’s invocation handler, passing the proxy instance,a java.lang.reflect.Method object identifying the method that was invoked, and an array of type Object containing the arguments.  

        由此可以知道以上的猜测是正确的,同时也知道,proxy参数传递的即是代理类的实例。 

        为了方便说明,这里写一个简单的例子来实现动态代理。 

       

Java代码  
收藏代码

  1. //抽象角色(动态代理只能代理接口)  
  2. public interface Subject {  
  3.       
  4.     public void request();  
  5. }  

Java代码  
收藏代码

  1. //真实角色:实现了Subject的request()方法  
  2. public class RealSubject implements Subject{  
  3.       
  4.     public void request(){  
  5.         System.out.println(“From real subject.”);  
  6.     }  
  7. }  

Java代码  
收藏代码

  1. //实现了InvocationHandler  
  2. public class DynamicSubject implements InvocationHandler  
  3. {  
  4.     private Object obj;//这是动态代理的好处,被封装的对象是Object类型,接受任意类型的对象  
  5.   
  6.     public DynamicSubject()  
  7.     {  
  8.     }  
  9.   
  10.     public DynamicSubject(Object obj)  
  11.     {  
  12.         this.obj = obj;  
  13.     }  
  14.   
  15.     //这个方法不是我们显示的去调用  
  16.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable  
  17.     {  
  18.         System.out.println(“before calling “ + method);  
  19.   
  20.         method.invoke(obj, args);  
  21.   
  22.         System.out.println(“after calling “ + method);  
  23.   
  24.         return null;  
  25.     }  
  26.   
  27. }  

Java代码  
收藏代码

  1. //客户端:生成代理实例,并调用了request()方法  
  2. public class Client {  
  3.   
  4.     public static void main(String[] args) throws Throwable{  
  5.         // TODO Auto-generated method stub  
  6.   
  7.         Subject rs=new RealSubject();//这里指定被代理类  
  8.         InvocationHandler ds=new DynamicSubject(rs);  
  9.         Class<?> cls=rs.getClass();  
  10.           
  11.         //以下是一次性生成代理  
  12.           
  13.         Subject subject=(Subject) Proxy.newProxyInstance(  
  14.                 cls.getClassLoader(),cls.getInterfaces(), ds);  
  15.           
  16.         //这里可以通过运行结果证明subject是Proxy的一个实例,这个实例实现了Subject接口  
  17.         System.out.println(subject instanceof Proxy);  
  18.           
  19.         //这里可以看出subject的Class类是$Proxy0,这个$Proxy0类继承了Proxy,实现了Subject接口  
  20.         System.out.println(“subject的Class类是:”+subject.getClass().toString());  
  21.           
  22.         System.out.print(“subject中的属性有:”);  
  23.           
  24.         Field[] field=subject.getClass().getDeclaredFields();  
  25.         for(Field f:field){  
  26.             System.out.print(f.getName()+“, “);  
  27.         }  
  28.           
  29.         System.out.print(“\n”+“subject中的方法有:”);  
  30.           
  31.         Method[] method=subject.getClass().getDeclaredMethods();  
  32.           
  33.         for(Method m:method){  
  34.             System.out.print(m.getName()+“, “);  
  35.         }  
  36.           
  37.         System.out.println(“\n”+“subject的父类是:”+subject.getClass().getSuperclass());  
  38.           
  39.         System.out.print(“\n”+“subject实现的接口是:”);  
  40.           
  41.         Class<?>[] interfaces=subject.getClass().getInterfaces();  
  42.           
  43.         for(Class<?> i:interfaces){  
  44.             System.out.print(i.getName()+“, “);  
  45.         }  
  46.   
  47.         System.out.println(“\n\n”+“运行结果为:”);  
  48.         subject.request();  
  49.     }  
  50. }  

Xml代码  
收藏代码

  1. 运行结果如下:此处省略了包名,***代替  
  2. true  
  3. subject的Class类是:class $Proxy0  
  4. subject中的属性有:m1, m3, m0, m2,   
  5. subject中的方法有:request, hashCode, equals, toString,   
  6. subject的父类是:class java.lang.reflect.Proxy  
  7. subject实现的接口是:cn.edu.ustc.dynamicproxy.Subject,   
  8.   
  9. 运行结果为:  
  10. before calling public abstract void ***.Subject.request()  
  11. From real subject.  
  12. after calling public abstract void ***.Subject.request()  

PS:这个结果的信息非常重要,至少对我来说。因为我在动态代理犯晕的根源就在于将上面的subject.request()理解错了,至少是被表面所迷惑,没有发现这个subject和Proxy之间的联系,一度纠结于最后调用的这个request()是怎么和invoke()联系上的,而invoke又是怎么知道request存在的。其实上面的true和class $Proxy0就能解决很多的疑问,再加上下面将要说的$Proxy0的源码,完全可以解决动态代理的疑惑了。 

        从以上代码和结果可以看出,我们并没有显示的调用invoke()方法,但是这个方法确实执行了。下面就整个的过程进行分析一下: 

        从Client中的代码看,可以从newProxyInstance这个方法作为突破口,我们先来看一下Proxy类中newProxyInstance方法的源代码: 

Java代码  
收藏代码

  1. public static Object newProxyInstance(ClassLoader loader,  
  2.         Class<?>[] interfaces,  
  3.         InvocationHandler h)  
  4. throws IllegalArgumentException  
  5. {  
  6.     if (h == null) {  
  7.         throw new NullPointerException();  
  8.     }  
  9.   
  10.     /* 
  11.      * Look up or generate the designated proxy class. 
  12.      */  
  13.     Class cl = getProxyClass(loader, interfaces);  
  14.   
  15.     /* 
  16.      * Invoke its constructor with the designated invocation handler. 
  17.      */  
  18.     try {  
  19.            /* 
  20.             * Proxy源码开始有这样的定义: 
  21.             * private final static Class[] constructorParams = { InvocationHandler.class }; 
  22.             * cons即是形参为InvocationHandler类型的构造方法 
  23.            */  
  24.         Constructor cons = cl.getConstructor(constructorParams);  
  25.         return (Object) cons.newInstance(new Object[] { h });  
  26.     } catch (NoSuchMethodException e) {  
  27.         throw new InternalError(e.toString());  
  28.     } catch (IllegalAccessException e) {  
  29.         throw new InternalError(e.toString());  
  30.     } catch (InstantiationException e) {  
  31.         throw new InternalError(e.toString());  
  32.     } catch (InvocationTargetException e) {  
  33.         throw new InternalError(e.toString());  
  34.     }  
  35. }  

        Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)做了以下几件事. 

        (1)根据参数loader和interfaces调用方法 getProxyClass(loader, interfaces)创建代理类$Proxy0.$Proxy0类 实现了interfaces的接口,并继承了Proxy类. 

        (2)实例化$Proxy0并在构造方法中把DynamicSubject传过去,接着$Proxy0调用父类Proxy的构造器,为h赋值,如下: 

Java代码  
收藏代码

  1. class Proxy{  
  2.     InvocationHandler h=null;  
  3.     protected Proxy(InvocationHandler h) {  
  4.         this.h = h;  
  5.     }  
  6.     …  
  7. }  

        来看一下这个继承了Proxy的$Proxy0的源代码: 

Java代码  
收藏代码

  1. public final class $Proxy0 extends Proxy implements Subject {  
  2.     private static Method m1;  
  3.     private static Method m0;  
  4.     private static Method m3;  
  5.     private static Method m2;  
  6.   
  7.     static {  
  8.         try {  
  9.             m1 = Class.forName(“java.lang.Object”).getMethod(“equals”,  
  10.                     new Class[] { Class.forName(“java.lang.Object”) });  
  11.   
  12.             m0 = Class.forName(“java.lang.Object”).getMethod(“hashCode”,  
  13.                     new Class[0]);  
  14.   
  15.             m3 = Class.forName(“***.RealSubject”).getMethod(“request”,  
  16.                     new Class[0]);  
  17.   
  18.             m2 = Class.forName(“java.lang.Object”).getMethod(“toString”,  
  19.                     new Class[0]);  
  20.   
  21.         } catch (NoSuchMethodException nosuchmethodexception) {  
  22.             throw new NoSuchMethodError(nosuchmethodexception.getMessage());  
  23.         } catch (ClassNotFoundException classnotfoundexception) {  
  24.             throw new NoClassDefFoundError(classnotfoundexception.getMessage());  
  25.         }  
  26.     } //static  
  27.   
  28.     public $Proxy0(InvocationHandler invocationhandler) {  
  29.         super(invocationhandler);  
  30.     }  
  31.   
  32.     @Override  
  33.     public final boolean equals(Object obj) {  
  34.         try {  
  35.             return ((Boolean) super.h.invoke(this, m1, new Object[] { obj })) .booleanValue();  
  36.         } catch (Throwable throwable) {  
  37.             throw new UndeclaredThrowableException(throwable);  
  38.         }  
  39.     }  
  40.   
  41.     @Override  
  42.     public final int hashCode() {  
  43.         try {  
  44.             return ((Integer) super.h.invoke(this, m0, null)).intValue();  
  45.         } catch (Throwable throwable) {  
  46.             throw new UndeclaredThrowableException(throwable);  
  47.         }  
  48.     }  
  49.   
  50.     public final void request() {  
  51.         try {  
  52.             super.h.invoke(this, m3, null);  
  53.             return;  
  54.         } catch (Error e) {  
  55.         } catch (Throwable throwable) {  
  56.             throw new UndeclaredThrowableException(throwable);  
  57.         }  
  58.     }  
  59.   
  60.     @Override  
  61.     public final String toString() {  
  62.         try {  
  63.             return (String) super.h.invoke(this, m2, null);  
  64.         } catch (Throwable throwable) {  
  65.             throw new UndeclaredThrowableException(throwable);  
  66.         }  
  67.     }  
  68. }  

        接着把得到的$Proxy0实例强制转换成Subject,并将引用赋给subject。当执行subject.request()方法时,就调用了$Proxy0类中的request()方法,进而调用父类Proxy中的h的invoke()方法.即InvocationHandler.invoke()。 

PS:1、需要说明的一点是,Proxy类中getProxyClass方法返回的是Proxy的Class类。之所以说明,是因为我一开始犯了个低级错误,以为返回的是“被代理类的Class类”- -!推荐看一下getProxyClass的源码,很长=。= 

        2、从$Proxy0的源码可以看出,动态代理类不仅代理了显示定义的接口中的方法,而且还代理了java的根类Object中的继承而来的equals()、hashcode()、toString()这三个方法,并且仅此三个方法。 

Q:到现在为止,还有一个疑问,invoke方法中的第一个参数是Proxy的实例(准确说,最终用到的是$Proxy0的实例),但是有什么用呢?或者说,程序内是怎样显示出作用的? 

A:就本人目前的水平看来,这个proxy参数并没有什么作用,在整个动态代理机制中,并没有用到InvocationHandler中invoke方法的proxy参数。而传入的这个参数实际是代理类的一个实例。我想可能是为了让程序员在invoke方法中使用反射来获取关于代理类的一些信息吧。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/131910.html原文链接:https://javaforall.net

(1)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 2018阿里巴巴春季校招笔试题(Java开发)「建议收藏」

    2018阿里巴巴春季校招笔试题(Java开发)「建议收藏」第一题:编程题要求:编译器版本:Java1.8.0_66请使用标准输入输出(System.in,System.out);已禁用图形、文件、网络、系统相关的操作,如java.lang.Process,javax.swing.JFrame,Runtime.getRuntime;不要自定义包名称,否则会报错,即不要添加packageanswer之类的语句;您可以写很多个类,但是必须有一个…

    2025年10月10日
    4
  • jenkins自动触发构建_为什么触发器有定时问题

    jenkins自动触发构建_为什么触发器有定时问题前言跑自动化用例每次用手工点击jenkins出发自动化用例太麻烦了,我们希望能每天固定时间跑,这样就不用管了,坐等收测试报告结果就行。jenkins的定时任务是用的crontab语法定时构建语法

    2022年7月28日
    19
  • p6操作教程_pc6视频教学

    p6操作教程_pc6视频教学在开发的过程中,我们经常会遇到由于sql语句书写错误导致的bug,那么如何来解决这种困扰呢?如果方法执行完了可以打印出完整的sql语句,就可以方便我们判断执行的是否正确,所以我们希望有一个可以打印sql语句的插件。p6spy就是一款这样的工具,下面给大家介绍一下p6spy的使用。使用p6spy需要做以下三步:1. 导入jar包:将jar包复制到项目中去,记得要buildpath一下。我用…

    2022年10月5日
    3
  • http请求415错误Unsupported Media Type

    http请求415错误Unsupported Media Type王子乔每一个认真生活的人,都值得被认真对待王子乔每一个认真生活的人,都值得被认真对待王子乔每一个认真生活的人,都值得被认真对待http请求415错误UnsupportedMediaTy

    2022年7月1日
    22
  • linux双系统默认进入win10,win10 linux 双系统 默认win10启动

    linux双系统默认进入win10,win10 linux 双系统 默认win10启动双系统怎么设置Win10系统为默认启动系统。Win10一出来,我就去安装尝试了,在自己电脑上安装了Win10双系统。安装后Win10就是默认启动系统了,那么要想让以前的系统为默认系统怎么设置呢,今天我就给大家介绍下Win10双系统怎么设置默认启动系统。1、在Win10系统下同时按下组合键“Win+R”,打开运行工具,输入“msconfig”,然后点击确定,如下图所示。2、在系统设置界面上点击“引导…

    2022年7月24日
    13
  • SD/MMC 卡读写模块—SD/MMC 卡的外部物理接口

    SD/MMC 卡读写模块—SD/MMC 卡的外部物理接口转载地址:http://www.8951.com/book/jiao1n21.htmSD/MMC   卡是一种大容量(最大可达4GB)、性价比高、体积小、访问接口简单的存储卡。SD/MMC卡大量应用于数码相机、MP3机、手机、大容量存储设备,作为这些便携式设备的存储载体,它还具有低功耗、非易失性、保存数据无需消耗能量等特点。SD卡接口向下兼容MMC(MutliMediaCard

    2022年6月8日
    48

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号