代理模式 Java设计模式笔记

阅读更多
代理模式的作用:
      为一些对象提供一种代理关系,来控制对这个对象的访问,从而避免调用者对这个对象的直接调用,而起到中介、保护的作用。
代理模式的元素:
      代理存在于事物的“间接关系”之中,不可或缺的需要三种角色:抽象角色、代理角色、真实角色。这里边的抽象角色,是用来解释“代理行为关系”的。
代理模式的故事场景:
      时间回到三国,蜀魏两国掐起来了,诸葛亮又要搞曹操。战争这事儿上,除了武力值,更多的是玩智商,诸葛亮手底下的赵云啊、关兴啊、张苞啊要是直接去跟曹操磕,那根本不是个儿,弄不好直接就被逮去砍头了,所以啊,开战的时候,需要老谋深算的诸葛亮坐镇代理,一会儿让赵云去冲,一会儿让关兴去冲,一会儿让张苞去冲,歇会儿再让谁去冲,不是说六出岐山么,每次换着花样冲,才能让曹操烦不胜烦,头疼欲裂。
角色提取及代码实现:
      既然是两军对垒,诸葛亮赵云小关张等都是将军,都能发起冲锋,肯定要抽象出来一个将军类,我们用接口实现:
/**
 * @author Veiking
 * 定义一个将军接口,诸葛亮,赵云,小关张都是将军
 */
public interface General {
    //将军都会发起冲锋,进行攻击
    public void attacking(); 
}

    赵云是老同志,身经百战,模范将军
/**
 * @author Veiking
 * 定义一个赵云,子龙将军
 */
public class Zhaoyun implements General{
    @Override
    public void attacking() {
        System.out.println("我乃常山赵子龙!进攻...");
    }
}

    关兴、张苞,年少有为,都是将军
/**
 * @author Veiking
 * 定义关兴,小关将军;定义张苞,小张将军
 */
public class Guanxing implements General{
    @Override
    public void attacking() {
        System.out.println("我乃关兴!左勾拳,进攻...");
    }
}
public class Zhangbao implements General{
    @Override
    public void attacking() {
        System.out.println("我乃张苞!右勾拳,进攻...");
    }
}

    诸葛亮老成持重,足智多谋,是领军之将,但他毕竟上点年纪,体力不行,自己折腾不动,更多的功能是指挥作战,具体的冲锋陷阵,得别人来做。默认是赵云做贴身大将,当然,让谁来出战都行
/**
 * @author Veiking
 * 定义一个诸葛亮,诸葛孔明,神人也,喜欢扇扇子
 */
public class Zhugeliang implements General{
    //具体作战的将军
    General general;
    //默认构造函数,赵云,是革命老同志,比较靠谱,默认贴身大将
    public Zhugeliang(){
        general = new Zhaoyun();
    }
    //有时候赵云忙,也可以换个将军来做贴身大将军
    public Zhugeliang(General general){
        this.general = general;
    }
    //打仗不能只用一个将军,适当的可以换个人
    public void setGeneral(General general){
        this.general = general;
    }
    //这个是重点,所有将军都不例外,诸葛亮发起进攻
    @Override
    public void attacking() {
        System.out.println("我乃诸葛亮,别跑!进攻...");
        //当然,老诸葛自己肯定冲不动,冲锋陷阵这事儿,得让具体的将军来干
        this.general.attacking();
    }
    
    ...
    //孔明的个性,招牌动作,多冷都能扇 again and again ...
    public void playFan(){
        System.out.println("羽扇羽扇,帮我算算...");
    }
}

    好,蜀国的诸位将军已经安排妥当,接下来,就该到曹操的营盘上来闹事儿了
/**
 * @author Veiking
 * 曹操的领地,跑来闹事儿的地方
 * (这里曹操领地System.out.println的“旁白”,自然是曹操说的,注意)
 */
public class CaocaoPlace {
    public static void main(String[] args) {
        //众将听命...兵贵诡,军情机密,赵云小关张都猫起来,随时战备出战
        General zhaoyun = new Zhaoyun();
        General guanxing = new Guanxing();
        General zhangbao = new Zhangbao();
        //诸葛亮出现了,蜀军叫阵,大战在即
        Zhugeliang zhugeliang = new Zhugeliang();
        //情节需要,先摇下扇子
        zhugeliang.playFan();
        System.out.println("孔明?!");
        zhugeliang.attacking();
        System.out.println("孔明?!...原来是赵云啊,我挡!");
        //换关兴上
        zhugeliang.setGeneral(guanxing);
        zhugeliang.attacking();
        System.out.println("孔明?!...原来是小关啊,我再挡!");
        //换张苞上
        zhugeliang.setGeneral(zhangbao);
        zhugeliang.attacking();
        System.out.println("孔明?!...原来是小张啊,我再挡!");
        //再换关兴上
        zhugeliang.setGeneral(guanxing);
        zhugeliang.attacking();
        System.out.println("孔明?!...小关怎么又来了,再挡!");
        //再换赵云上
        zhugeliang.setGeneral(zhaoyun);
        zhugeliang.attacking();
        System.out.println("孔明?!...擦,怎么又...受不了了...");
        //情节需要,扇子不能停...
        zhugeliang.playFan();
        ...
        //换了一拨又一拨...
        System.out.println("孔明?!...擦,头痛...");
        ...
        //冲了一次又一次...
        System.out.println("孔明?!...头痛欲裂...");
        //注意,这里边各位蜀将若是不自报家门,曹司令至死都不知道跟自己磕的是谁
    }
}

      据说,诸葛亮是这样去找曹操麻烦,整整六次,让曹头痛不已,最后抢救无效,薨。
小结:
      这里边诸葛亮做的是蜀国将军大代理,有效的避免暴露手下将军智力方面短板,又发挥了他们骁勇善战的长处,算是一举两得,并且无论哪位将军有事儿忙,随时可替补更换,不影响整个事情的发展;从程序的角度来说,代理类这边处理好对外业务,真实类处理好具体事物行为,只要都对接口负责就可以了。

=====================================================
代理模式++ 动态代理:
      代理模式中我们使用代理角色来代理一类真实角色,如果要代理的真实角色种类非常多,可以是将军也可以是小兵,甚至还可以是军马战畜,这时候一个代理类是不能满足所需功能的,代码量可想而知。事实上,诸葛亮管好将军们已经实属不易,如果还去管小兵、车马,那可就要早早累死了。于是“动态代理”横空出世,java引入的动态代理机制,完美的让诸葛亮晋级军师,成为一代大神。
代码实现:
      想发动“动态代理”大招,要引入这三个重要类InvocationHandler、Method、Proxy,其功能暂不表述,且听下回分说,先看诸葛孔明如何做些许改动
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/** 
 * @author Veiking 
 * 定义一个动态诸葛亮,这个诸葛亮不简单,更像个军事
 */ 
public class DynamicZhugeliang implements InvocationHandler {
    private Object obj;
    //一看,参数都Object类了,基本可以包罗万象,什么都可以传进来了
    DynamicZhugeliang(Object obj) {
        this.obj = obj;
    }
    //被代理实际对象的方法代理,这绝对操心的到位,扇子摇一摇,啥事都搞定
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object result = null;
        //出招前后,扇子要摇一摇……
        System.out.println("before doing 。。。 羽扇羽扇,帮我算算...");   
        result = method.invoke(obj, args);//
        System.out.println("after doing 。。。 羽扇羽扇,帮我算算...");
        return result;
    }
    //对传进来的任何真实对象,都进行动态生成相应的代理对象,孔明有了这招,几近无敌!
    public static Object proxy(Object obj) {
        Class clazz = obj.getClass();
        return Proxy.newProxyInstance(clazz.getClassLoader(), clazz.getInterfaces(), new DynamicZhugeliang(obj));
    }
}

      曹操的领地也要做一点点改动
/**
 * @author Veiking 曹操的动态领地,跑来闹事儿的地方 (旁白还是帮曹操说的)
 */
public class DynamicCaocaoPlace {
    public static void main(String[] args) throws Throwable {
        //将旗招展,大写一个“将”字,是谁出战骚扰,不知道
        General general = null;
        //众将听命...兵贵诡,军情机密,赵云小关张都猫起来,随时战备出战  
        General zhaoyun = new Zhaoyun();  
        General guanxing = new Guanxing();  
        General zhangbao = new Zhangbao();
        //注意诸葛亮在这时候,已经不惜的出来露脸了,只是默默的摇扇子做他的大代理,指挥各位将军轮番闹阵(当然,催催粮草啊,治理国家啊,只要有具体角色提取,都基本不是事儿了)……
        general = (General) DynamicZhugeliang.proxy(zhaoyun);
        general.attacking();
        System.out.println("谁?!...原来是赵云啊,我挡!"); 
        general = (General) DynamicZhugeliang.proxy(guanxing);
        general.attacking();
        System.out.println("谁?!...原来是小关啊,我再挡!");  
        general = (General) DynamicZhugeliang.proxy(zhangbao);
        general.attacking();		
        System.out.println("谁?!...原来是小张啊,我再挡!");  
        ...
        //可怜曹司令,又是一阵头疼……
    }
}

小结:
      话不多说,仅凭诸葛亮的factory方法的参数Object就知道,他已经逆天了,其中动态代理的奥妙,下次笔记细聊。

(故事纯属瞎构,代码仅供参考)

你可能感兴趣的:(java,代理模式,动态代理,设计模式,Spring)