动态代理的好处:
1.代理方式多样,自由定义,比如可以查看被代理类的各方法执行时间。
2.当被代理对象改变其内部实现时,不影响代理规则。
动态代理的局限性:
1.只能代理interface方法
以拦截Button的onClick方法为例说明下我们的动态代理模式:
1.首先创建一个OnClickListener 的实现类
class DefaultClickListener implements View.OnClickListener {
@Override
public void onClick(View view) {
Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
.setAction("Action", null).show();
}
}
2.然后开始生成该实现类的代理类
DefaultClickListener defaultClickListener = new DefaultClickListener();
Object proxyClickListener = Proxy.newProxyInstance(fab.getClass().getClassLoader(), defaultClickListener.getClass().getInterfaces(), new ClickListenerInvocationHandler(defaultClickListener));
Proxy.newProxyInstance方法的第三个参数InvocationHandler就是我们需要的调用处理类,以下是一个简单的实现:
class ClickListenerInvocationHandler implements InvocationHandler{
private Object defaultObject;
public ClickListenerInvocationHandler(Object defaultObject){
this.defaultObject = defaultObject;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if("onClick".equals(method.getName())){
Toast.makeText(MainActivity.this,"filter onClick method",Toast.LENGTH_SHORT).show();
method.invoke(defaultObject,args);
}
return null;
}
}
该实现类中的invoke方法,我们判断调用的方法是否是onClick方法,如果是则执行我们的拦截处理,此处打印了一句话,并继续执行被代理类的方法。
3.给button设置clickListener为新的代理类
button.setOnClickListener((View.OnClickListener) proxyClickListener);