说到代理,我们日常生活中有很多例子可以与之类比:我们去银行取钱,有业务员帮我们处理,我们去买饭,卖饭的人帮我们把饭打包好等。
代理又分为动态代理和静态代理。下面我先讲解静态代理。
1,静态代理
静态代理使用去银行办卡为场景来写Demo:
银行办卡角色划分:
(1)目标接口
(2)代理对象——银行业务员
(3)被代理对象——我
以下是具体的例子,注释写在代码中,因此就不再多罗嗦:
/*
* 银行现在只有一项业务
* */
public interface IBank {
public void applyBank();
}
/*
* 这是代理对象,也就是银行的业务员,他对银行的业务肯定十分了解
* */
public class BankWorker implements IBank{
private IBank bank;
//在这里银行的业务员需要持有被代理人对象
public BankWorker(IBank bank) {
this.bank=bank;
}
public void applyBank() {
System.out.println("开始办理");
bank.applyBank();
System.out.println("办理结束");
}
}
/*
* 被代理对象
* */
public class Man implements IBank{
private String name;
public Man(String name) {
this.name=name;
}
public void applyBank() {
System.out.println(this.name+"来银行办理业务");
}
}
测试代码:
public class TestDemo{
public static void main(String args[]) {
IBank man=new Man("Bob");
IBank bankWorker=new BankWorker(man);
bankWorker.applyBank();
}
}
输出:
开始办理
Bob来银行办理业务
办理结束
接着讲解动态代理:
上卖弄静态代理的缺点很明显:当在银行的业务中添加一项新的业务的时候,也就是再银行接口中新添加一个方法的时候,其余的实现了银行接口的类都要重新实现这个方法,而且这些方法实现起来代码大致类似,因此我们考虑使用动态代理:
/*
* 银行现在只有一项业务
* */
public interface IBank {
public void applyBank();
}
/*
* 被代理对象
* */
public class Man implements IBank{
private String name;
public Man(String name) {
this.name=name;
}
public void applyBank() {
System.out.println(this.name+"来银行办理业务");
}
}
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
/*
* 银行办理业务,动态代理
* */
public class BankInvocationHandler implements InvocationHandler{
private Object mObject;//被代理的对象
public BankInvocationHandler(Object object) {
this.mObject=object;
}
@Override
public Object invoke(Object arg0, Method arg1, Object[] arg2) throws Throwable {
//执行方法,目标接口调用的方法都会来到这里面
System.out.println("开始办理业务");
//调用被代理对象的方法,也就是man的方法
Object voidObject=arg1.invoke(mObject, arg2);
System.out.println("操作完毕");
return voidObject;
}
}
//测试代码
import java.lang.reflect.Proxy;
public class TestDemo{
public static void main(String args[]) {
IBank man=new Man("Bob");
//参数1:类加载器,参数2:目标接口,参数3:InvocationHandler
//返回的是IBank的一个实例,这个对象最终是由java给我们创建的,调用的是JNI
IBank bank=(IBank)Proxy.newProxyInstance(IBank.class.getClassLoader(), new Class<?>[] {IBank.class},
new BankInvocationHandler(man));
//当调用这个方法的时候会执行BankInvocationHandler的invoke()方法
bank.applyBank();
}
}