Java代理模式
1. 简介
代理模式(Proxy Pattern)是GoF 23种Java常用设计模式之一。代理模式的定义:Provide a surrogate or placeholder for another object to controlaccess to it(为其他对象提供一种代理以控制对这个对象的访问)。使用代理模式创建代理对象,让代理对象控制目标对象的访问(目标对象可以是远程的对象、创建开销大的对象或需要安全控制的对象),并且可以在不改变目标对象的情况下添加一些额外的功能。
2. UML类图
3. 模式中包含的角色及其职责
Subject:抽象主题角色,抽象主题类可以是抽象类,也可以是接口,是一个最普通的业务类型定义,无特殊要求。
RealSubject:具体主题角色,也叫被委托角色、被代理角色。是业务逻辑的具体执行者。
Proxy:代理主题角色,也叫委托类、代理类。它把所有抽象主题类定义的方法给具体主题角色实现,并且在具体主题角色处理完毕前后做预处理和善后工作。(最简单的比如打印日志)
4. 代码实现
(1) Subject
package com.jackie.designpatterns.proxy;
/**
* 抽象主题,定义主要功能
*/
publicinterface Subject {
publicvoid operate();
}
(2) RealSubject
package com.jackie.designpatterns.proxy;
/**
* 具体主题
*/
publicclass RealSubject implements Subject{
@Override
publicvoid operate() {
System.out.println("realsubject operatestarted......");
}
}
(3) Proxy
package com.jackie.designpatterns.proxy;
/**
* 代理类
*/
publicclass Proxy implements Subject{
private Subject subject;
public Proxy(Subject subject) {
this.subject = subject;
}
@Override
publicvoid operate() {
System.out.println("before operate......");
subject.operate();
System.out.println("after operate......");
}
}
(4) Client
package com.jackie.designpatterns.proxy;
/**
* 客户
*/
publicclass Client {
/**
* @param args
*/
publicstaticvoid main(String[] args) {
Subject subject = new RealSubject();
Proxy proxy = new Proxy(subject);
proxy.operate();
}
}
运行结果: beforeoperate......
realsubject operate started......
afteroperate......
5. 应用场景
现实世界中,秘书就相当于一个代理,老板开会,那么通知员工开会时间、布置会场、会后整理会场等等开会相关工作就可以交给秘书做,老板就只需要开会就行了,不需要亲自做那些事。同理,在我们程序设计中也可使用代理模式来将由一系列无关逻辑组合在一起的代码进行解耦合,比如业务代码中的日志代码就可以在代理中进行。Spring的AOP就是典型的动态代理应用。
6. 代理模式的应用形式
(1)远程代理(Remote Proxy) -可以隐藏一个对象存在于不同地址空间的事实。也使得客户端可以访问在远程机器上的对象,远程机器可能具有更好的计算性能与处理速度,可以快速响应并处理客户端请求。
(2)虚拟代理(Virtual Proxy) – 允许内存开销较大的对象在需要的时候创建。只有我们真正需要这个对象的时候才创建。
(3)写入时复制代理(Copy-On-Write Proxy) – 用来控制对象的复制,方法是延迟对象的复制,直到客户真的需要为止。是虚拟代理的一个变体。
(4)保护代理(Protection (Access)Proxy) – 为不同的客户提供不同级别的目标对象访问权限
(5)缓存代理(Cache Proxy) – 为开销大的运算结果提供暂时存储,它允许多个客户共享结果,以减少计算或网络延迟。
(6)防火墙代理(Firewall Proxy) – 控制网络资源的访问,保护主题免于恶意客户的侵害。
(7)同步代理(SynchronizationProxy) – 在多线程的情况下为主题提供安全的访问。
(8)智能引用代理(Smart ReferenceProxy) - 当一个对象被引用时,提供一些额外的操作,比如将对此对象调用的次数记录下来等。
(9)复杂隐藏代理(Complexity HidingProxy) – 用来隐藏一个类的复杂集合的复杂度,并进行访问控制。有时候也称为外观代理(Façade Proxy),这不难理解。复杂隐藏代理和外观模式是不一样的,因为代理控制访问,而外观模式是不一样的,因为代理控制访问,而外观模式只提供另一组接口。
【代理模式应用场景举例】
比如在玩“极品飞车”这款游戏,如果游戏者手中的金钱达到了一定的数量就可以到车店买一部性能更高的赛车,那么这个卖车的“车店”就是一个典型的“汽车厂家”的“代理”,他为汽车厂家“提供卖车的服务”给有需求的人士。从面向对象的方面考虑,“销售汽车的代理”也是一个对象,那么这个对象也具有一定的状态,在软件项目中这个对象也具有管理财务进销存的基本功能,那么在设计时就要以面向OOP编程的思想来考虑软件的类结构,这个销售汽车的代理也是一个类了。
【代理模式解释】
类型:结构模式
对一些对象提供代理,以限制那些对象去访问其它对象。
【代理模式-JAVA代码实现】
新建一个买车的接口:
- public interface buy_car {
- public void buy_mycar();
- }
新建一个people人类,具有买车的行为,所以实现接口buy_car
- class people implements buy_car {
- private int cash;
- private String vip;
- private String username;
- @Override
- public void buy_mycar() {
- // TODO Auto-generated method stub
- System.out.print(username+"是vip 客户,可以直接购买新车!");
- }
- public int getCash(){
- return cash;
- }
- public void setCash(int cash){
- this.cash = cash;
- }
- public String getUsername(){
- return username;
- }
- public void setUsername(String username){
- this.username = username;
- }
- public String getVip(){
- return vip;
- }
- public void setVip(String vip){
- this.vip = vip;
- }
- }
people类不能拥有车,必须经过proxy代理类的认证,符合条件之后才可以拥有车辆,新建一个代理,这个代理类来考察当前的people是否有资格进行买车:
- class proxyclass implements buy_car {
- private people people;
- public people getPeople(){
- return people;
- }
- public void setPeople(people people){
- this.people = people;
- }
- @Override
- public void buy_mycar() {
- // TODO Auto-generated method stub
- if (people.getVip() == "vip"){
- people.buy_mycar();
- return ;
- }
- if(people.getCash()>=50000){
- System.out.println(people.getUsername()+"买了新车,交易结束!");
- }
- else
- {
- System.out.println(people.getUsername()+"钱不够,不能买车,继续比赛!");
- }
- }
- }
最后创建一个客户端,用来模拟买车的行为:
- public class run_main {
- public static void main(String[] args) {
- // TODO Auto-generated method stub
- people people_1 =new people();
- people_1.setCash(60000);
- people_1.setUsername("jeck");
- people people_2 =new people();
- people_2.setCash(40000);
- people_2.setUsername("rose");
- people people_3 =new people();
- people_3.setCash(0);
- people_3.setUsername("tom");
- people_3.setVip("vip");
- proxyclass proxy_buy = new proxyclass();
- proxy_buy.setPeople(people_1);
- proxy_buy.buy_mycar();
- proxy_buy.setPeople(people_2);
- proxy_buy.buy_mycar();
- proxy_buy.setPeople(people_3);
- proxy_buy.buy_mycar();
- }
- }
程序运行结果如下:
- jeck买了新车,交易结束!
- rose钱不够,不能买车,继续比赛!
- tom是vip 客户,可以直接购买新车!
代理模式主要使用了java的多态,干活的是被代理类,代理类主要是接活,你让我干活,好,我交给幕后的类去干,你满意就成,那怎么知道被代理类能不能干呢?同根就成,大家知根知底,你能做啥,我能做啥都清楚得很,同样一个接口。