2 动态代理生成的代理类到底是什么样子的?生成代理类的关键接口是什么?为何调用真实对象的某个接口会进入invoke方法?
通过分析源码:我们知道 Proxy.newProxyInstance → Proxy.getProxyClass0 → WeakCache.get → WeakCache.Factory.get → Proxy.ProxyClassFactory.apply → ProxyGenerator.generateProxyClass
最终生成了一个 byte[] 类型的 class类;这样byte[] 比较抽象 ,我们想看到该怎么办?可以通过下边的代码生成Proxy0
int accessFlags = Modifier.PUBLIC | Modifier.FINAL; byte[] bytes = ProxyGenerator.generateProxyClass("com.sun.proxy.$Proxy0", new Class[]{IAnimal.class}, accessFlags); FileOutputStream fileOutputStream = new FileOutputStream(new File( "~/work/$Proxy0.class")); fileOutputStream.write(bytes); fileOutputStream.flush(); fileOutputStream.close();
生成的代码如下:
// // Source code recreated from a .class file by IntelliJ IDEA // (powered by Fernflower decompiler) // package com.sun.proxy; import com.shock.base.proxy.dynamic.IAnimal; 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 IAnimal { 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 boolean equals(Object var1) throws { try { return ((Boolean)super.h.invoke(this, m1, new Object[]{var1})).booleanValue(); } catch (RuntimeException | Error var3) { throw var3; } catch (Throwable var4) { throw new UndeclaredThrowableException(var4); } } public final String toString() throws { try { return (String)super.h.invoke(this, m2, (Object[])null); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } } public final void bark() throws { try { super.h.invoke(this, m3, (Object[])null); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } } public final int hashCode() throws { try { return ((Integer)super.h.invoke(this, m0, (Object[])null)).intValue(); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } } static { try { m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[]{Class.forName("java.lang.Object")}); m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]); m3 = Class.forName("com.shock.base.proxy.dynamic.IAnimal").getMethod("bark", new Class[0]); m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]); } catch (NoSuchMethodException var2) { throw new NoSuchMethodError(var2.getMessage()); } catch (ClassNotFoundException var3) { throw new NoClassDefFoundError(var3.getMessage()); } } }
以上代码为我们解答了红色的问题。虽然代码是这样的 代理类集成了 Proxy类,但是如果想要验证 如何验证呢?
cglibcglib是什么?CGLIB is a powerful, high performance code generation library.
特点简单说:
CGLib (Code Generation Library) 是一个强大的,高性能,高质量的Code生成类库;
它可以在运行期扩展Java类与实现Java接口;
CGLib 比 Java 的 java.lang.reflect.Proxy 类更强的在于它不仅可以接管接口类的方法,还可以接管普通类的方法。
CGLib 的底层是Java字节码操作框架 —— ASM
引入JAR包支持,如下:
<dependency> <groupId>cglib</groupId> <artifactId>cglib</artifactId> <version>3.2.4</version> </dependency>
目前最新的版本是 3.2.4,CGLib的package分布和作用如下:
net.sf.cglib.core:底层字节码处理类,他们大部分与ASM有关系,对其进行封装,更易于使用;
net.sf.cglib.transform:编译期或运行期类和类文件的转换;
net.sf.cglib.proxy:实现创建代理和方法拦截器的类;
net.sf.cglib.reflect:实现快速反射的类;
net.sf.cglib.util:集合排序工具类;
net.sf.cglib.beans:JavaBean相关的工具类;
我们沿用上边的例子,来做下演示:
加入你有一个类Animal,打算对里边的所有方法进行包装,由于这个类没有实现接口,所以你无法使用jdk 动态代理