jdk原生动态代理:
// 接口
interface Hello{
String sayHello(String str);
}
// 实现
class HelloImp implements Hello{
@Override
public String sayHello(String str) {
return "HelloImp: " + str;
}
}
// Java Proxy
// 1. 首先实现一个InvocationHandler,方法调用会被转发到该类的invoke()方法。
class LogInvocationHandler implements InvocationHandler{
...
private Hello hello;
public LogInvocationHandler(Hello hello) {
this.hello = hello;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if("sayHello".equals(method.getName())) {
logger.info("You said: " + Arrays.toString(args));
}
return method.invoke(hello, args);
}
}
// 2. 然后在需要使用Hello的时候,通过JDK动态代理获取Hello的代理对象。
Hello hello = (Hello)Proxy.newProxyInstance(
getClass().getClassLoader(), // 1. 指定代理对象的类加载器
new Class<?>[] {Hello.class}, // 2. 代理对象需要实现的接口,可以有多个
new LogInvocationHandler(new HelloImp()));// 3. 方法调用的实际处理者,代理对象的方法调用都会转发到这里
System.out.println(hello.sayHello("I love you!"));
newProxyInstance()会返回一个实现了指定接口的代理对象,对该对象所有方法的条用都会转发给InvocationHandler.invoke()方法。
代理对象的类型是jdkProxy.$Proxy0,父类是java.lang.reflect.Proxy,所有jdk代理都会实现这个类
jdk动态代理是基于接口的
====================================================================================================================================================================
cglib动态代理:
cglib无法代理final方法,因为cglib通过继承实现代理,final类型的类无法被继承,遇到final类型的类会抛异常(java.lang.IllegalArgumentException: Cannot subclass final class cglib.HelloConcrete),final类型的方法也无法被重写
public class HelloConcrete {
public String sayHello(String str) {
return "HelloConcrete: " + str;
}
}
// CGLIB动态代理
// 1. 首先实现一个MethodInterceptor,方法调用会被转发到该类的intercept()方法。
class MyMethodInterceptor implements MethodInterceptor{
...
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
logger.info("You said: " + Arrays.toString(args));
return proxy.invokeSuper(obj, args);
}
}
// 2. 然后在需要使用HelloConcrete的时候,通过CGLIB动态代理获取代理对象。
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(HelloConcrete.class);
enhancer.setCallback(new MyMethodInterceptor());
HelloConcrete hello = (HelloConcrete)enhancer.create();
System.out.println(hello.sayHello("I love you!"));