1、前言
今天技术分享时,有人分享了 Springboot 应用中事务失败的例子,如类中有 A、B 两个方法,B 用 @Transaction 注解标识,但是在 A 中调用 B 方法时 B 方法的事务不会起作用。原因是调用 A 方法时不会走代理类的切片,然后说到了动态代理、反射等知识。至于这里的详细原因,下次专门开一篇文章分享,进行我想捋清楚的时反射和动态代理的关系。
我们都知道反射可以获取一个对象的所以信息,包括字段、方法等,也可以直接获得该类的 Method 对象,直接 invoke 调用累对象的方法,类似于这样。反射的原理我们不讲,就讲这个用法。
Class<?> clazz = Q1_reverseList.class;
Q1_reverseList q1_reverseList = (Q1_reverseList) clazz.newInstance();
clazz.getMethod("aa").invoke(q1_reverseList);
那么动态代理与反射又有什么关系呢?其实主要的关系在于生成的代理类调用被代理类(原对象)的方法那块,其他真的没了。可以看看👇这个生成的代理类。
public final class $Proxy0 extends Proxy implements ICook {
private static Method m1;
private static Method m2;
private static Method m3;
private static Method m4;
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 cook() throws {
try {
super.h.invoke(this, m3, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
public final void dealWithFoot() throws {
try {
super.h.invoke(this, m4, (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.company.ICook").getMethod("cook", new Class[0]);
m4 = Class.forName("com.company.ICook").getMethod("dealWithFoot", 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());
}
}
}
当调用 cook() 方法时,是调用 InvocationHandler 实现类的 invoke 方法,而此方法就是通过反射 invoke 被代理类的方法来调用目标方法。
说到这里,我们讨论了一个最简单的需求。假设一个类有很多方法,如果想为这个类所有的方法增加一个计时功能,那么思路肯定是在方法调用前记录一个 startTime,调用结束后记录一个 endTime,然后相减就是耗时。怎样才能不重复写代码?
开始我想到用静态代理,但是问题是我得实现你所有的方法,然后在方法前后重复写计时。而且想不到不重复写的思路。
而用动态代理,我只需要在 InvocationHandler 实现类的 invoke 方法中写上计时逻辑,然后在中间调用目标方法即可,非常简单。运行结果如下图:


网友评论