原文
Spring学习(五):动态代理的两种实现方式(全网最容易懂)_P@ssW0rd的博客-CSDN博客
1、基于JDK的动态代理
基于接口的动态代理,用到的类是Proxy的newProxyInstance静态方法创建,要求被代理对象至少实现一个接口,如果没有,则不能创建代理对象。
2、基于cglib的动态代理
要导入cglib第三方库,使用的类是Enhancer的create静态方法创建,要求被代理类不能是最终类,即不能用final修饰,如String类。
三、代码演示
1、首先创建一个IProduct接口,并创建被代理类,实现这个接口
IProduct
public interface IProduct {
String sell(Float money);
void afterSell();
}
public class Product implements IProduct {
@Override
public String sell(Float money) {
System.out.println("代理员交给工厂:"+money);
return "aaa";
}
@Override
public void afterSell() {
System.out.println("代理员做售后。。");
}
}
通过JDK来实现动态代理,创建一个消费者Consumer
这里我们直接通过匿名内部类来实现,当然不是必须的
public class Consumer {
public static void main(String[] args) {
// 创建一个被代理对象
final Product product = new Product();
// 创建一个代理对象,并在InvocationHandler的invoke方法里面,对被代理类的方法做增强
IProduct proxyProduct = (IProduct) Proxy.newProxyInstance(product.getClass().getClassLoader(), product.getClass().getInterfaces(), new InvocationHandler() {
// 实现具体的增强操作
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 获取方法在运行中可能产生的返回值
Object returnValue = null;
Float money = (Float) args[0];
if("sell".equals(method.getName())){
// 执行具体的方法
returnValue = method.invoke(product, money*0.8F);
}
return returnValue;
}
});
System.out.println(proxyProduct.sell(1000F));
}
}
IProduct proxyProduct = (IProduct) Proxy.newProxyInstance(product.getClass().getClassLoader(), product.getClass().getInterfaces(), new InvocationHandler() {
}
ClassLoader loader获取被代理类的类加载器。
Class<?>[] interfaces获取被代理类的实现接口的数组。
InvocationHandler h在invok方法中对方法做增强处理。
invoke方法的三个参数
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
}
3、通过cglib来实现动态代理,创建一个消费者Consumer
public class Consumer {
public static void main(final String[] args) {
// 创建一个被代理对象,这里要求必须是final
final Product product = new Product();
Product proxyProduct =(Product) Enhancer.create(product.getClass(), new MethodInterceptor() {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
Float money = (Float) objects[0];
Object returnValue = null;
if("sell".equals(method.getName())){
returnValue = method.invoke(product, 0.8f * money);
}
return returnValue;
}
});
System.out.println(proxyProduct.sell(1000f));
}
}
Enhancer.create的2个参数
Product proxyProduct =(Product) Enhancer.create(product.getClass(), new MethodInterceptor() {
}
Class type被代理类的class文件
Callback callback一个Callback接口,我们通常使用MethodInterceptor接口,继承了Callback接口
ntercept方法的参数
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
}
Method method当前方法
Object[] objects方法用到的参数数组
网友评论