2.cglb动态代理是基于代理类生成其子类来实现的.
首先看这个被代理类,接下来需要对study方法进行逻辑增强
public class Tom {
String stduy(String name){
System.out.println("原生方法");
return name;
}
}
实现接口MethodInterceptor在intercept中完成逻辑的增强,
public class MyMethodInterceptor implements MethodInterceptor {
/**
* 代理对象会调用此方法
* @param o 代理的对象.
* @param method 被代理对象执行的方法
* @param objects 被代理的对象的方法入参
* @param methodProxy
* @return 返回值就是被代理方法执行后的返回值.这里必须要返回.否则原生方法执行完后没有返回值
* @throws Throwable
*/
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println(method.getName());
//执行被代理的方法
System.out.println("原生方法执行前");
Object invoke = methodProxy.invokeSuper(o, objects);
System.out.println("原生方法执行后");
return invoke ;
}
}
代理工厂,用于获取代理类的对象.
public class CglbFactroy<T> {
public T getProxy(T o){
Enhancer enhancer = new Enhancer();
//设置方法拦截.即方法增强的逻辑
enhancer.setCallback(new MyMethodInterceptor());
//设置谁需要被代理
enhancer.setSuperclass(o.getClass());
//创建一个代理对象并且返回
return (T)enhancer.create();
}
}
测试:
这里面加了一段代码
System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY,"d://cglb_classes");
这样的话我们就可以去指定位置查看到cglb到底生成了个啥.
public class Test {
public static void main(String[] args) {
//设置源码的位置,可以看到代理类的生成情况
System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY,"d://cglb_classes");
Tom tom = new Tom();
CglbFactroy<Tom> tomCglbFactroy = new CglbFactroy<>();
Tom proxy = tomCglbFactroy.getProxy(tom);
proxy.stduy("name");
}
}
执行后去d盘查看:
注意这里面可能会生成多个类,最后找到这个:
拖拽到idea打开:
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by FernFlower decompiler)
//
package cglbProxy;
import java.lang.reflect.Method;
import net.sf.cglib.core.ReflectUtils;
import net.sf.cglib.core.Signature;
import net.sf.cglib.proxy.Callback;
import net.sf.cglib.proxy.Factory;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class Tom$$EnhancerByCGLIB$$8bcc110c extends Tom implements Factory {
private boolean CGLIB$BOUND;
public static Object CGLIB$FACTORY_DATA;
private static final ThreadLocal CGLIB$THREAD_CALLBACKS;
private static final Callback[] CGLIB$STATIC_CALLBACKS;
private MethodInterceptor CGLIB$CALLBACK_0;
private static Object CGLIB$CALLBACK_FILTER;
private static final Method CGLIB$stduy$0$Method;
private static final MethodProxy CGLIB$stduy$0$Proxy;
private static final Object[] CGLIB$emptyArgs;
private static final Method CGLIB$equals$1$Method;
private static final MethodProxy CGLIB$equals$1$Proxy;
private static final Method CGLIB$toString$2$Method;
private static final MethodProxy CGLIB$toString$2$Proxy;
private static final Method CGLIB$hashCode$3$Method;
private static final MethodProxy CGLIB$hashCode$3$Proxy;
private static final Method CGLIB$clone$4$Method;
private static final MethodProxy CGLIB$clone$4$Proxy;
static void CGLIB$STATICHOOK1() {
CGLIB$THREAD_CALLBACKS = new ThreadLocal();
CGLIB$emptyArgs = new Object[0];
Class var0 = Class.forName("cglbProxy.Tom$$EnhancerByCGLIB$$8bcc110c");
Class var1;
CGLIB$stduy$0$Method = ReflectUtils.findMethods(new String[]{"stduy", "(Ljava/lang/String;)Ljava/lang/String;"}, (var1 = Class.forName("cglbProxy.Tom")).getDeclaredMethods())[0];
CGLIB$stduy$0$Proxy = MethodProxy.create(var1, var0, "(Ljava/lang/String;)Ljava/lang/String;", "stduy", "CGLIB$stduy$0");
Method[] var10000 = ReflectUtils.findMethods(new String[]{"equals", "(Ljava/lang/Object;)Z", "toString", "()Ljava/lang/String;", "hashCode", "()I", "clone", "()Ljava/lang/Object;"}, (var1 = Class.forName("java.lang.Object")).getDeclaredMethods());
CGLIB$equals$1$Method = var10000[0];
CGLIB$equals$1$Proxy = MethodProxy.create(var1, var0, "(Ljava/lang/Object;)Z", "equals", "CGLIB$equals$1");
CGLIB$toString$2$Method = var10000[1];
CGLIB$toString$2$Proxy = MethodProxy.create(var1, var0, "()Ljava/lang/String;", "toString", "CGLIB$toString$2");
CGLIB$hashCode$3$Method = var10000[2];
CGLIB$hashCode$3$Proxy = MethodProxy.create(var1, var0, "()I", "hashCode", "CGLIB$hashCode$3");
CGLIB$clone$4$Method = var10000[3];
CGLIB$clone$4$Proxy = MethodProxy.create(var1, var0, "()Ljava/lang/Object;", "clone", "CGLIB$clone$4");
}
final String CGLIB$stduy$0(String var1) {
return super.stduy(var1);
}
public final String stduy(String var1) {
MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
if (var10000 == null) {
CGLIB$BIND_CALLBACKS(this);
var10000 = this.CGLIB$CALLBACK_0;
}
return var10000 != null ? (String)var10000.intercept(this, CGLIB$stduy$0$Method, new Object[]{var1}, CGLIB$stduy$0$Proxy) : super.stduy(var1);
}
final boolean CGLIB$equals$1(Object var1) {
return super.equals(var1);
}
public final boolean equals(Object var1) {
MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
if (var10000 == null) {
CGLIB$BIND_CALLBACKS(this);
var10000 = this.CGLIB$CALLBACK_0;
}
if (var10000 != null) {
Object var2 = var10000.intercept(this, CGLIB$equals$1$Method, new Object[]{var1}, CGLIB$equals$1$Proxy);
return var2 == null ? false : (Boolean)var2;
} else {
return super.equals(var1);
}
}
final String CGLIB$toString$2() {
return super.toString();
}
public final String toString() {
MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
if (var10000 == null) {
CGLIB$BIND_CALLBACKS(this);
var10000 = this.CGLIB$CALLBACK_0;
}
return var10000 != null ? (String)var10000.intercept(this, CGLIB$toString$2$Method, CGLIB$emptyArgs, CGLIB$toString$2$Proxy) : super.toString();
}
final int CGLIB$hashCode$3() {
return super.hashCode();
}
public final int hashCode() {
MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
if (var10000 == null) {
CGLIB$BIND_CALLBACKS(this);
var10000 = this.CGLIB$CALLBACK_0;
}
if (var10000 != null) {
Object var1 = var10000.intercept(this, CGLIB$hashCode$3$Method, CGLIB$emptyArgs, CGLIB$hashCode$3$Proxy);
return var1 == null ? 0 : ((Number)var1).intValue();
} else {
return super.hashCode();
}
}
final Object CGLIB$clone$4() throws CloneNotSupportedException {
return super.clone();
}
protected final Object clone() throws CloneNotSupportedException {
MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
if (var10000 == null) {
CGLIB$BIND_CALLBACKS(this);
var10000 = this.CGLIB$CALLBACK_0;
}
return var10000 != null ? var10000.intercept(this, CGLIB$clone$4$Method, CGLIB$emptyArgs, CGLIB$clone$4$Proxy) : super.clone();
}
public static MethodProxy CGLIB$findMethodProxy(Signature var0) {
String var10000 = var0.toString();
switch(var10000.hashCode()) {
case -508378822:
if (var10000.equals("clone()Ljava/lang/Object;")) {
return CGLIB$clone$4$Proxy;
}
break;
case 6566728:
if (var10000.equals("stduy(Ljava/lang/String;)Ljava/lang/String;")) {
return CGLIB$stduy$0$Proxy;
}
break;
case 1826985398:
if (var10000.equals("equals(Ljava/lang/Object;)Z")) {
return CGLIB$equals$1$Proxy;
}
break;
case 1913648695:
if (var10000.equals("toString()Ljava/lang/String;")) {
return CGLIB$toString$2$Proxy;
}
break;
case 1984935277:
if (var10000.equals("hashCode()I")) {
return CGLIB$hashCode$3$Proxy;
}
}
return null;
}
public Tom$$EnhancerByCGLIB$$8bcc110c() {
CGLIB$BIND_CALLBACKS(this);
}
public static void CGLIB$SET_THREAD_CALLBACKS(Callback[] var0) {
CGLIB$THREAD_CALLBACKS.set(var0);
}
public static void CGLIB$SET_STATIC_CALLBACKS(Callback[] var0) {
CGLIB$STATIC_CALLBACKS = var0;
}
private static final void CGLIB$BIND_CALLBACKS(Object var0) {
Tom$$EnhancerByCGLIB$$8bcc110c var1 = (Tom$$EnhancerByCGLIB$$8bcc110c)var0;
if (!var1.CGLIB$BOUND) {
var1.CGLIB$BOUND = true;
Object var10000 = CGLIB$THREAD_CALLBACKS.get();
if (var10000 == null) {
var10000 = CGLIB$STATIC_CALLBACKS;
if (var10000 == null) {
return;
}
}
var1.CGLIB$CALLBACK_0 = (MethodInterceptor)((Callback[])var10000)[0];
}
}
public Object newInstance(Callback[] var1) {
CGLIB$SET_THREAD_CALLBACKS(var1);
Tom$$EnhancerByCGLIB$$8bcc110c var10000 = new Tom$$EnhancerByCGLIB$$8bcc110c();
CGLIB$SET_THREAD_CALLBACKS((Callback[])null);
return var10000;
}
public Object newInstance(Callback var1) {
CGLIB$SET_THREAD_CALLBACKS(new Callback[]{var1});
Tom$$EnhancerByCGLIB$$8bcc110c var10000 = new Tom$$EnhancerByCGLIB$$8bcc110c();
CGLIB$SET_THREAD_CALLBACKS((Callback[])null);
return var10000;
}
public Object newInstance(Class[] var1, Object[] var2, Callback[] var3) {
CGLIB$SET_THREAD_CALLBACKS(var3);
Tom$$EnhancerByCGLIB$$8bcc110c var10000 = new Tom$$EnhancerByCGLIB$$8bcc110c;
switch(var1.length) {
case 0:
var10000.<init>();
CGLIB$SET_THREAD_CALLBACKS((Callback[])null);
return var10000;
default:
throw new IllegalArgumentException("Constructor not found");
}
}
public Callback getCallback(int var1) {
CGLIB$BIND_CALLBACKS(this);
MethodInterceptor var10000;
switch(var1) {
case 0:
var10000 = this.CGLIB$CALLBACK_0;
break;
default:
var10000 = null;
}
return var10000;
}
public void setCallback(int var1, Callback var2) {
switch(var1) {
case 0:
this.CGLIB$CALLBACK_0 = (MethodInterceptor)var2;
default:
}
}
public Callback[] getCallbacks() {
CGLIB$BIND_CALLBACKS(this);
return new Callback[]{this.CGLIB$CALLBACK_0};
}
public void setCallbacks(Callback[] var1) {
this.CGLIB$CALLBACK_0 = (MethodInterceptor)var1[0];
}
static {
CGLIB$STATICHOOK1();
}
}
可以看到这里是集成了Tom类的,并且重写了方法stduy:
public final String stduy(String var1) {
MethodInterceptor var10000 = this.CGLIB$CALLBACK_0;
if (var10000 == null) {
CGLIB$BIND_CALLBACKS(this);
var10000 = this.CGLIB$CALLBACK_0;
}
return var10000 != null ? (String)var10000.intercept(this, CGLIB$stduy$0$Method, new Object[]{var1}, CGLIB$stduy$0$Proxy) : super.stduy(var1);
}
关键的一段代码:
(String)var10000.intercept(this, CGLIB$stduy$0$Method, new Object[]{var1}, CGLIB$stduy$0$Proxy)
这里就是在调用前面类
MyMethodInterceptor
中重写的方法.
至此就可以发现完整的代理类就出来了.细心的小伙伴发现了,如果方法是final修饰或者private修饰是不能被代理的哦