定义
代理模式为另一个对象提供一个替身或占位符以控制对这个对象的访问
**
- 使用代理模式创建代表对象, 让代表对象控制某对象的访问, 被代理的对象可以是远程的对象, 创建开销大的对象或需要安全控制的对象
- 远程代理管理客户和远程对象之间的交互
- 虚拟代理控制访问实例化开销大的对象
- 保护代理基于调用者控制对对象方法的访问
- 代理模式有很多变体例如: 缓存代理, 同步代理, 防火墙代理, 和写入时复制代理等

远程代理
案例
糖果机需要一个可以远程监控的监控机, 通过代理模式, java rmi实现
虚拟代理


保护代理
- 动态代理之所以被称为动态, 是因为运行时才将他的类创建出来, 代码开始执行时, 还没有proxy类, 它是根据需要从你传入的接口集创建的
利用java的动态代理来实现保护代理
案例
背景: 只能给别人评分, 可以设置自己的姓名
PersonBean
public interface PersonBean {String getName();void setName(String name);int getRate();void setRate(int rate);}
PersonBeanImpl
public class PersonBeanImpl implements PersonBean {private String name;private int rate;public PersonBeanImpl(String name, int rate) {this.name = name;this.rate = rate;}@Overridepublic String getName() {return name;}@Overridepublic void setName(String name) {this.name = name;}@Overridepublic int getRate() {return rate;}@Overridepublic void setRate(int rate) {this.rate = rate;}}
OwnerInvocationHandler
public class OwnerInvocationHandler implements InvocationHandler {PersonBean person;public OwnerInvocationHandler(PersonBean person) {this.person = person;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {try {if (method.getName().equals("setName") || method.getName().startsWith("get")) {return method.invoke(person, args);} else {throw new IllegalAccessException();}} catch (InvocationTargetException e) {e.printStackTrace();}return null;}}
测试
@Testpublic void testProxy() {PersonBeanImpl personBean = new PersonBeanImpl("小王", 9);PersonBean ownerProxy = getOwnerProxy(personBean);System.out.println(ownerProxy.getName());System.out.println(ownerProxy.getRate());ownerProxy.setName("小张");System.out.println(ownerProxy.getName());try {ownerProxy.setRate(1);} catch (Exception e) {System.out.println("不能设置自己的分数!");}}private PersonBean getOwnerProxy(PersonBean person) {return (PersonBean) Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(),new OwnerInvocationHandler(person));}}测试结果:小王9小张不能设置自己的分数!


