动态代理
-
动态代理是根据运行时的反射机制来实现.代理类是动态生成的
-
类似于代购:代理类只做代理一件事情,无论上下游是谁
- 好处是代理类代理的是接口,一般就是一类业务:如中间商代理雅诗兰黛的化妆品
- 另外代理类可以代理多个类,只要实现接口就可以了:中间商可以有很多客户,只要客户要的是雅诗兰黛的化妆品
-
动态代理分为两大类:基于接口JDK动态代理(常用)
-
基于类:BGLib
-
还有一种JAVAssist,由java字节码实现的,由东京大学研究出来.用于JBoos服务器
接口类
package com.li.changGe.designMode.dynamicProxy;
public interface ScriptKill {
void buy();
}
用户类实现接口的方法
package com.li.changGe.designMode.dynamicProxy;
public class User implements ScriptKill {
private String name;
public User(String name){
this.name = name;
}
public void buy(){
System.out.println(name+"喜欢剧本杀");
}
}
代理类实现InvocationHandler接口的invoke()
InvocationHandler调用处理:每个代理实例都有一个关联的调用处理程序
-
Proxy.newProxyInstance()调用时会自动调用这个方法
-
这个Method对象理解成上游接口
-
我们运行要被实现方法的接口的方法,就是调用实现类的方法
-
得到的返回值对象也就是实现类了,
-
method.invoke()理解成:实现类.newInstance
-
而参数就是接口方法需要的参数:有就传,没有不传
-
可以对代理方法进行增强,比如在代理方法执行前后添加代码
package com.li.changGe.designMode.dynamicProxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
/**
* 不要在大家听不懂的地方停留太久
*/
public class Dynamic implements InvocationHandler {
private Object target;
public Dynamic(Object target){
this.target = target;
}
//------------------------------------------------------------
//把这两个方法的传参,一一对应会好理解一些
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
/**
* Proxy.newProxyInstance()调用时会自动调用这个方法
*
* 这个Method对象理解成上游接口
* 我们运行要被实现方法的接口的方法,就是调用实现类的方法
*
* 得到的返回值对象也就是实现类了,
* method.invoke()理解成:实现类.newInstance
*
* 而参数就是接口方法需要的参数:有就传,没有不传
*/
//可以对代理方法进行增强,比如在代理方法执行前后添加代码
System.out.println(method.getName()+"方法被执行了");
Object invoke = method.invoke(target, args);
System.out.println(method.getName()+"方法执行完了");
return invoke;
}
}
Test测试
-
我们调用newProxyInstance()时,底层会自动调用InvocationHandler的invoke().
-
并一一对应的传参过去
- 三个传参分别是类加载器,接口数组和InvocationHandler对象
- 类加载器就是要运行哪个类的方法,传入用户类还是接口的都可以
- 而接口数组,因为用户类实现了接口,所以用户类就是接口类型.而接口是没法获取接口数组的(你自己能获取相同的自己吗)
- 代理类实现了InvocationHandler,所以代理类就是InvocationHandler类型,而我们要调用的invoke()是写在代理类中,所以要传入代理类
package com.li.changGe.designMode.dynamicProxy;
import java.lang.reflect.Proxy;
/**
* 共有三个类:接口,用户和代理类
*/
public class Test {
public static void main(String[] args) {
User user = new User("长歌");
Dynamic dynamic = new Dynamic(user);
/**
* 我们调用newProxyInstance()时,底层会自动调用InvocationHandler的invoke().
* 并一一对应的传参过去
*/
ScriptKill proxy =(ScriptKill)Proxy.newProxyInstance(user.getClass().getClassLoader(),User.class.getInterfaces()
, dynamic);
proxy.buy();
}
}