1.Cglib动态代理
目标类:
package cn.happy.proxy.cglib;
public class Service {
public Service() {
System.out.println("创建Service对象");
}
public void doWordk(){
System.out.println("do something");
}
}
测试:
package cn.happy.proxy.cglib;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
public class CglibTest {
public static void main(String[] args) {
//对方法增强。开闭原则-对修改关闭,对添加放开
//1.创建一个目标对象
Service service=new Service();
//2.Enhancer对象
Enhancer enhancer=new Enhancer();
enhancer.setSuperclass(service.getClass());
//3.调度setCallBack()
enhancer.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("write log");
methodProxy.invoke(service,objects);
return null;
}
});
//4.Enhancer对象的Create()方法创建代理
Service proxy =(Service) enhancer.create();
//5.执行代理方法
proxy.doWordk();
}
}
测试结果:
测试第27行enhancer.create()方法调用了Service的构造器。
create()方法是生成代理对象,其本质是目标类的子类实例,所以会调用父类构造。
2.JDK动态代理
jdk动态代理原理是动态生成目标对象实现接口的实现类。
接口:
package cn.happy.proxy.jdk;
public interface IService {
public void doWordk();
}
目标类:
package cn.happy.proxy.jdk;
public class ServiceImpl implements IService{
public ServiceImpl() {
System.out.println("创建Service对象");
}
@Override
public void doWordk(){
System.out.println("do something");
}
}
测试:
package cn.happy.proxy.jdk;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class JdkTest {
public static void main(String[] args) {
ServiceImpl service = new ServiceImpl();
IService proxy = (IService)Proxy.newProxyInstance(service.getClass().getClassLoader(), service.getClass().getInterfaces(), new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("write log");
method.invoke(service, args);
return null;
}
});
proxy.doWordk();
}
}
测试结果:
3.比较
这里从网上摘了点儿东西
JDK动态代理:代理类和目标类必须实现相同的接口,客户端通过代理类来调用目标方法,代理类会将所有的方法调用分派到目标对象上反射执行,还可以在分派过程中添加"前置通知"和后置处理。
CGLIB动态代理:动态生成一个要代理类的子类,子类重写要代理的类的所有不是final的方法。在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势织入横切逻辑。它比使用java反射的JDK动态代理要快。
手机扫一扫
移动阅读更方便
你可能感兴趣的文章