字节码增强技术
aop方案
在aop方法中常见有ASM、aspectJ、JDKProxy、Javaassist几种
提示:以下是本篇文章正文内容,下面案例可供参考
一、ASM
cglib是基于ASM上实现的,ASM常用于自己操作字节码的需求,因为它可以在被类加载之前动态地修改生成,也可以直接生成.class文件在放入jvm。
ASM api是通过ClassWriter 接口基于访问者模式,让我们使用ClassVisitor,MethodVisitor,FieldVisitor API接口进行回调实现。
ASM是在汇编指令层次级别上操作字节码的,需要使用者有一定的class组织结构和jvm汇编基础。
二、java assist
JBoos旗下的一个产品,可以直接通过java编程的方式动态生成类或者改变类结构,操作简单。
三、JDKProxy
JDKProxy的实现原理
通过使用Proxy类中的newProxyInstance方法生成一个proxy代理类,并且是实现InvocationHandler回调接口来调用方法,但是想要代理的对象需要自身实现接口才能实现这种方式。
为什么动态代理需要实现接口呢?因为java不允许双继承,在反编译下代理对象 proxy0本身需要继承Proxy这个类,然后通过implments接口,来与想要代理的类产生联系,通过接口中的方法来实现代理。
//继承InvocationHandler
public class Invocation implements InvocationHandler {
private Object object;
public void setServiceObject(Object object) {
this.object = object;
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//执行方法前
System.out.println("before");
//执行方法
Object returnObject = method.invoke(this.object, args);
//执行方法后
System.out.println("after");
return returnObject;
}
}
//代理对象需要的接口接口
public interface Service {
public void dodo();
}
//代理对象 implements Service 这个接口
public class ServiceImpl implements Service {
public final void dodo() {
System.out.println("doing");
}
}
public class start {
public static void main(String[] args) {
Invocation invocation = new Invocation();
invocation.setServiceObject(new ServiceImpl());
Service service = (Service) Proxy.newProxyInstance(start.class.getClassLoader(),
new Class[]{Service.class}, invocation);
service.dodo();
}
}
/** 这时候生成的代理类
public final class $Proxy0 extends Proxy implements Service{
....
}
结果
before
doing
after
**/
jdk动态代理和cglib的选择
在spring中,当进行动态代理时,如果代理对象有实现接口,那么默认使用jdkproxy方法,没有的时候使用cglib方式。在jdk1.6之前,基于字节码技术生成的cglib相对于jdk动态代理的反射要快,1.6后经过优化后,如果不存在大量调用方法的情况下,jdk动态代理比cglib要快了,如今1.8及之后,jdk动态代理已经完全快于cglib了。