代理模式

代理模式定义:为其他对象提供一种代理以控制对这个对象的访问.

举个栗子:小明在外打工,临近春节抢不到票,于是找到黄牛,最后黄牛'神通广大',帮他买到一张回家的火车票。

这里面黄牛就是代理角色,买票是抽象请求角色,小明是具体请求者角色。

代码清单-1 买票接口

public interface ITicketRequest {
    /**
     * 买票请求
     */
    void buyTicket();
}

代码清单-2  工人类

public class Workman implements ITicketRequest {
    private String name ;
    public Workman(String name){
        this.name = name;
    }
    
    public void buyTicket() {
        System.out.println(name+",正在买票...");
    }
    
    //此处省略name的get和set方法...
}

代码清单-3 收费接口(黄牛收费)

public interface IPay {
    /**
     * 车票代购计费
     */
    void count();
}

代码清单-4 黄牛类

public class HuangNiu implements ITicketRequest,IPay{
    private ITicketRequest ticketRequest = null;
    public  HuangNiu(ITicketRequest ticketRequest) {
        this.ticketRequest = ticketRequest;
    }


    public void buyTicket() {
        //代替小明买票
        ticketRequest.buyTicket();
        //代理的个性方法
        count();
    }
    
    public void count() {
        System.out.println("买票费用一共是: 150元");
    }
}

代码清单-5 场景类 

public class Client {

    public static void main(String[] args){
        ITicketRequest xiaoming = new Workman("小明");
        HuangNiu huangNiu = new HuangNiu(xiaoming);
        //黄牛帮小明代购火车票并收费150元
        huangNiu.buyTicket();
    }
}

以上类图关系如下:

以上为静态代理模式.

继续上面的例子,小明发现买错了票,想要退票或者改签,这时需要在"黄牛"类里面新增退票和改签方法了,那如果小明后续还需要其他功能,这个"黄牛"代理就要做新增方法改动。那有没有办法让被代理类新增方法后,代理类不需要改动呢?答案是肯定的。

动态代理模式

代码清单-6  黄牛类 -动态代理

public class HuangNiu implements InvocationHandler {
    private Object ticketRequest = null;
    public  HuangNiu(Object ticketRequest) {
        this.ticketRequest = ticketRequest;
    }


    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("*** DynamicProxyPatterns start***");
        //代替小明买票,退票,改签等操作
        method.invoke(ticketRequest, args);
        System.out.println("*** DynamicProxyPatterns end ***");
        return null;
    }

    public Object createProxyInstance(){
        ClassLoader workmanClassLoader = ticketRequest.getClass().getClassLoader();
        Class<?>[] iTicketRequestInterface = ticketRequest.getClass().getInterfaces();
        return Proxy.newProxyInstance(workmanClassLoader,iTicketRequestInterface,this);
    }

}

代码清单-7 场景类

public class Client {

    public static void main(String[] args){
        Workman xiaoming = new Workman("小明");
        HuangNiu huangNiu = new HuangNiu(xiaoming);
        //动态创建一个代理类
        ITicketRequest xiaomingProxy = (ITicketRequest)huangNiu.createProxyInstance();
        //代理类实现买票方法
        xiaomingProxy.buyTicket();
    }
}

执行结果

*** DynamicProxyPatterns start***
小明,正在买票...
*** DynamicProxyPatterns end ***

 

posted @ 2016-07-01 00:56  Keeper丶  阅读(142)  评论(0编辑  收藏  举报