spring 系列6 动态代理
在业务中使用动态代理,一般是为了给需要实现的方法添加预处理或者添加后续操作,但是不干预实现类的正常业务,把一些基本业务和主要的业务逻辑分离。
动态代理
特点
字节码随用随创建,随用随加载
分类
基于接口的动态代理
基于子类的动态代理
一、基于接口的动态代理
使用JDK官方提供的Proxy
如何创建代理对象
使用Proxy类中的newProxyInstance方法
创建代理对象的要求
被代理类最少实现一个接口,如果没有则不能使用
newProxyInstance方法的参数
ClassLoader:代理类的类加载器,用于加载代理对象字节码
Class[]:代理类要实现的接口列表
InvocationHandler:用于提供增强的代码,它是让我们写如何代理。我们一般都是些一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的,此接口的实现类都是谁用谁写
1、创建一个接口
public interface ISubject{
void hello(String param);
}
2、实现接口
public class SubjectImpl implements ISubject{
public void hello(String param){
System.out.println("Hello " + param);
}
}
3、创建代理类
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class SubjectProxy implements InvocationHandler {
private ISubject subject;
public SubjectProxy(ISubject subject){
this.subject = subject;
}
/**
* 作用:执行被代理对象的任何接口方法都会经过该方法
* @param proxy 代理对象的引用
* @param method 当前执行的方法
* @param args 当前执行方法所需的参数
* @return 被代理对象方法的返回值
* @throws Throwable
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
System.out.println("begin");
Object invoke = method.invoke(subject, args);
System.out.println("end");
return invoke;
}
}
4、测试
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
public class ProxyTest {
public static void main(String[] args) {
ISubject target = new SubjectImpl();
ISubject proxyInstance = (ISubject) Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), new SubjectProxy(target));
proxyInstance.hello("mantishell");
}
}
在被代理之后,实际调用的方法是SubjectProxy的invoke方法,这样可以在不修改业务类的情况下对业务类增加一些日志等操作。
二、基于子类的动态代理
需要用到第三方cglib库
如何创建代理对象
使用Enhancer类中的create方法
注意:被代理类不能是最终类
create方法的参数:
Class:字节码,用于指定被代理对象的字节码
Callback:用于提供增强代码