Java动态代理,底层详细解析

1.动态代理需要了解的知识

类反射机制

Reflection(反射)是被视为动态语言的关键,反射机制允许程序在执行期借助于Reflection API取得任何类的内部信息,并能直接操作任意对象的内部属性及方法。

实例化Class类对象
1、前提:若已知具体的类,通过类的class属性获取,该方法最为安全可靠,程序性能最高

Class cl = String.class;

2、前提:已知某个类的实例,调用该实例的getClass()方法获取Class对象

Class cl ="abcabc".getClass();

3、前提:已知一个类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取,可能抛出 ClassNotFoundException

Class cla= Class.forName("java.lang.String");

了解以下几个包下的类

  • java.lang.Class:代表一个类
  • java.lang.reflect.Method:代表类的方法
  • java.lang.reflect.Field:代表类的成员变量
  • java.lang.reflect.Constructor:代表类的构造方法

2.动态代理例子

要用到的几类和方法在如下包:

-java.lang.reflect.proxy
-java.lang.reflect.InvocationHandler


1.public interface InvocationHandler
在这里插入图片描述
参数
loader :类加载器来定义代理类
interfaces : 代理类实现的接口列表
h :调度方法调用的调用处理函数


动态代理模拟图
在这里插入图片描述

1.Job接口

//抽象角色(要被代理的)
public interface Job {
    public void request();
}

2.Job的实现类

//真实角色
public class Company implements Job {
    public void request() {
        System.out.println("要招聘人才");
    }
}

3.动态生成类

用这个类自动生成代理类
public class ProxyInvocationHandler implements InvocationHandler {
    //Setter被代理的接口
    private Job job;
    public void setJob(Job job){
        this.job=job;
    }
    //核心就在这个方法了,它返回的是一个代理对象
    public Object getProxy(){
        Job proxy = (Job)Proxy.newProxyInstance(this.getClass().getClassLoader(), job.getClass().getInterfaces(), this);
        System.out.println(proxy.getClass().toString());
        return proxy;
    }

    //处理代理实例,执行Job的方法,也就是说实现“要招聘人才”需求
    //重写InvocationHandler,invoke的方法
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("before calling " + method);
        method.invoke(job, args);
        System.out.println("after calling " + method);
        return null;
    }
}

测试类

public class Client {
    public static void main(String[] args) {
   		 //真实角色,
        Company company = new Company();
        pih.setJob(company);
        //实例化动态生成代理的类
         ProxyInvocationHandler pih = new ProxyInvocationHandler();
        //动态生成代理
    //System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
        Job proxy = (Job)pih.getProxy();
        proxy.request();
    }
}

结果:
在这里插入图片描述
只看第三行,输出"要招聘人才",确实通过代理模式把它输出了。其他不用看是我测试用的

3.疑惑解析

1.关于newProxyInstance()的三个参数

Proxy.newProxyInstance(this.getClass().getClassLoader(), 
job.getClass().getInterfaces(), this);

1.this.getClass().getClassLoader()这个是类加载器,用来定义代理类

拓展:因为每个编写的”.java”拓展名类文件都存储着需要执行的程序逻辑,这些”.java”文件经过Java编译器编译成拓展名为”.class”的文件,”.class”文件中保存着Java代码经转换后的虚拟机指令,当需要使用某个类时,虚拟机将会加载它的”.class”文件,并创建对应的class对象。而将class文件加载到虚拟机的内存,这个过程称为类加载
(如果在深入就要去了解Java虚拟机)

2.job.getClass().getInterfaces()就是获得这个接口

3.因为第三个参数需要传InvocationHandler对象,而我们当前类继承了这个接口,所以用this

2.由始至终没有调用invoke()方法,而会走它呢?

可以看到,在测试类中或者生成代理类中,我们并没有显示的调用invoke()方法。
1.我们newProxyInstance这个方法作为突破口,查看源代码如下

public static Object newProxyInstance(ClassLoader loader,
                                      Class<?>[] interfaces,
                                      InvocationHandler h)
    throws IllegalArgumentException
{
    Objects.requireNonNull(h);

    final Class<?>[] intfs = interfaces.clone();
    final SecurityManager sm = System.getSecurityManager();
    if (sm != null) {
        checkProxyAccess(Reflection.getCallerClass(), loader, intfs);
    }
    /*
     * Look up or generate the designated proxy class.
     */
    Class<?> cl = getProxyClass0(loader, intfs);

    /*
     * Invoke its constructor with the designated invocation handler.
     */
    try {
        if (sm != null) {
            checkNewProxyPermission(Reflection.getCallerClass(), cl);
        }

        final Constructor<?> cons = cl.getConstructor(constructorParams);
        final InvocationHandler ih = h;
        if (!Modifier.isPublic(cl.getModifiers())) {
            AccessController.doPrivileged(new PrivilegedAction<Void>() {
                public Void run() {
                    cons.setAccessible(true);
                    return null;
                }
            });
        }
        return cons.newInstance(new Object[]{h});
}

看点:
1.Class<?> cl = getProxyClass0(loader, intfs);
2.final Constructor<?> cons = cl.getConstructor(constructorParams)

看点1中,泛型不用管,我们没有用到,这个两个参数loader,intfs就是我们刚刚传的类加载和接口(抽象角色)并且返回一个Class类对象cl,这个类就是我们要的创建代理类。

其实这个代理类的真实类名叫$Proxy0,在代码中我已经将这个代理类名输出了。
class com.sun.proxy.$Proxy0

这个.Clss类(所谓代理类)是Java虚拟机帮我们生成的,在虚拟机内存。我们现在把这个.class文件输出,反编译后,继续查看底层代码

//如果想看$Proxy0源码,可以在测试类上加这句话
//.class文件会生成在idea/当前项目/com/下
//(不是用Idea自己找,差不多,idea可以帮我们将.class文件反编译)
System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");

看点:
1.构造方法
2.方法 public final void request()

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

    public $Proxy0(InvocationHandler var1) throws  {
        super(var1);
    }
......

    public final void request() throws  {
        try {
            super.h.invoke(this, m3, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }
 ...........
 

当执行proxy.request()方法时,就调用了$Proxy0类中的request()方法,进而调用父类Proxy中的h的invoke()方法.即InvocationHandler.invoke()。就是代理类中实现的接口下的方法哦。

如果还不是不懂,继续看下面~~~
(这还是没return代理类对象前的那段代码…)

final Constructor<?> cons = cl.getConstructor(constructorParams);

将实例化$Proxy0并在构造方法中把类名为(ProxyInvocationHandler,这个类实现了InvocationHandler接口)传过去,接着$Proxy0调用父类Proxy的构造器,为h赋值,如下:

class Proxy{  
    InvocationHandler h=null;  
    protected Proxy(InvocationHandler h) {  
        this.h = h;  
    }  
    ....
}  

3.为什么用Job接收代理类对象

在查看完 $Proxy0 源码,就知道啦,它实现了Job接口,并且继承了Proxy 类

public final class $Proxy0 extends Proxy implements Job{
}

4.区分method.invoke()和重写的public Object invoke()

1.method.invoke()是method类下的方法:
public final class Method extends Executable {
public Object invoke(Object obj, Object… args)
}
2.InvocationHandler接口下的方法
public interface InvocationHandler {
public Object invoke(Object proxy, Method method, Object[] args)
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值