动态代理 - zhpanvip/AndroidNote GitHub Wiki
建议先了解静态代理
1.静态代理的缺点
-
由于代理类要实现与被代理类一致的接口,当有多个类需要被代理时,要么代理类实现所有被代理类的接口,这样会使代理类过于庞大;要么使用多个代理类,每个代理类只代理一个被代理类,但是这样又会需要构造多个代理类。
-
当接口需要增加、删除、修改方法时,被代理类与代理类都需要修改,不易维护。
为了解决上述问题,可以使用动态代理,自动生成代理类。
2.使用JDK提供的接口实现动态代理
这里涉及到两个类:java.lang.reflect.Proxy和
java.lang.reflect.InvocationHandler接口
举个例子:
Ryan想在上海买一套房子,但是他又不懂房地产的行情,于是委托了中介(Proxy)来帮助他买房子。
使用动态代理实现,首先定义一个IPersonBuyHouse的接口,且有一个buyHouse的方法:
public interface IPersonBuyHouse {
// 购买房子的方法,返回值代表是否成功购买
boolean buyHouse(String name);
}
使用动态代理实现Ryan买房子的需求:
public static void main(String[] args) {
InvocationHandler handler = new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 在执行购买逻辑前可以先做一些校验,对于不符合要求的不予执行,并返回false。这里省略不写了...
if (method.getName().equals("buyHouse")) {
System.out.println(args[0] + " will buy a house.");
}
// 返回true,表示成功购买
return true;
}
};
IPersonBuyHouse person = (IPersonBuyHouse) Proxy
.newProxyInstance(IPersonBuyHouse.class.getClassLoader(), // ClassLoader
new Class[]{IPersonBuyHouse.class}, // 传入要实现的接口
handler); // 传入处理调用方法的InvocationHandler
person.buyHouse("Ryan");
}
2. 动态代理原理
动态代理是在运行时根据某个接口生成对应的代理类的字节码,然后加载到JVM的,并创建这个接口的实例的过程。
对于第二节中的例子进行分析,通过Proxy.newProxyInstance这个方法会生成一个IPersonBuyHouse的实现类。假设生成的这个类叫Ryan,Ryan这个类实现了IPersonBuyHouse,并重写了buyHouse的方法。在这个buyHouse的方法中会调用InvocationHandler的invoke方法。
Ryan类的代码大致如下:
public class Ryan implements IPersonBuyHouse {
// buyHouse的真正逻辑在这个匿名内部类的invoke方法中
private InvocationHandler handler = new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) {
// 被代理类真正执行的逻辑
if (method.getName().equals("buyHouse")) {
System.out.println(args[0] + " will buy a house.");
}
return true;
}
};
@Override
public boolean buyHouse(String name) {
try {
// 反射获取buyHouse这个Method
Method buyHouseMethod = IPersonBuyHouse.class.getDeclaredMethod("buyHouse", String.class);
// 将buyHouse的参数封装成一个数组
Object[] params = new Object[1];
params[0] = name;
// 实际调用了InvocationHandler的invoke方法
return (boolean) handler.invoke(this, buyHouseMethod, params);
} catch (Throwable e) {
e.printStackTrace();
}
return false;
}
}