首頁 >Java >java教程 >Java設計模式之代理模式的筆記

Java設計模式之代理模式的筆記

黄舟
黄舟原創
2017-10-18 09:39:001194瀏覽

這篇文章主要為大家詳細介紹了java設計模式之代理模式筆記,具有一定的參考價值,有興趣的小夥伴們可以參考一下

代理程式(Proxy)模式:

代理模式是物件的結構模式。代理模式提供某一個物件一個代理對象,並由代理對象控制原對象的參考。

核心作用:

透過代理,控制對物件的存取。

可以詳細控制存取某個(某一類別)物件的方法,在呼叫這個方法之前做前置處理,在呼叫這個方法之後做後置處理。即AOP的微觀實現。

AOP(Aspect Oriented Programming面向切面程式設計)的核心實作機制。

生活中的場景:

  所謂代理,就是一個人或機構代表另一個人或機構採取行動。在某些情況下,一個客戶不想或無法直接引用一個對象,而代理對象可以在客戶端和目標對象之間起到中介的作用。
  例如,客戶要找某個明星唱歌,首先需要找到他的經紀人,然後由他的經紀人來安排某明星唱歌。
  經紀人在演唱會舉辦前需要處理一些前置處理(面談,合約起草,簽字,收預付款,安排機票和車輛等)在演唱會舉辦後需要處理一些後置處理(收尾款等) 。此時某位明星(真實角色)只需要關心如何唱歌就可以了,其他事情全部交給經紀人(代理)來辦。

核心角色:

這裡寫圖片描述

抽象物件角色:宣告了代理物件和真實物件的共同接口,定義代理物件和真實物件的公共對外方法。這樣一來在任何可以使用真實物件的地方都可以使用代理物件。

真實物件角色:定義了代理物件所代表的真實物件。實作抽象對象,定義真實對象所需實現的業務邏輯,供代理對象呼叫。專注於真正的業務邏輯。

代理對象角色:實作抽象對象,是真實對象的代理,透過真實對象的業務邏輯方法來實作抽象方法,並附加自己的操作。將統一的流程控制放到代理對像中處理。

  代理物件內部含有真實物件的引用,從而可以在任何時候操作真實物件;代理物件提供一個與真實物件相同的接口,以便在任何時候都可以取代真實物件。代理物件通常在客戶端呼叫傳遞給真實物件之前或之後,執行某個操作,而不是單純地將呼叫傳遞給真實物件。

應用程式場景:

安全代理程式:屏蔽對真實角色的直接存取。
遠端代理:透過代理類別處理遠端方法呼叫。
延遲載入:先載入輕量級的代理對象,真正需要再去載入真實對象。 (圖片的延遲載入)

分類:

靜態代理程式:(靜態定義代理類別)

上面範例的程式碼:

1、宣告了代理物件和真實物件的共同接口,定義代理物件和真實物件的公共對外方法。


public interface Star {
  /**
   * 面谈
   */
  void confer();
  /**
   * 签合同
   */
  void signContract();
  /**
   * 订票
   */
  void bookTicket();
  /**
   * 唱歌
   */
  void sing();
  /**
   * 收钱
   */
  void collectMoney();
}

2、定義一個真實物件的類,實作抽象介面所提供的方法。


public class RealStar implements Star {

  @Override
  public void bookTicket() {
    System.out.println("RealStar.bookTicket()");
  }

  @Override
  public void collectMoney() {
    System.out.println("RealStar.collectMoney()");
  }

  @Override
  public void confer() {
    System.out.println("RealStar.confer()");
  }

  @Override
  public void signContract() {
    System.out.println("RealStar.signContract()");
  }

  @Override
  public void sing() {
    System.out.println("RealStar.sing()");
  }
}

3、定義一個代理物件的類,實作抽象介面所提供的方法,並持有真實物件的參考。


public class ProxyStar implements Star{

private Star star;

  public ProxyStar(Star star) {
    super();
    this.star = star;
  }

  @Override
  public void bookTicket() {
    System.out.println("ProxyStar.bookTicket()");
  }

  @Override
  public void collectMoney() {
    System.out.println("ProxyStar.collectMoney()");
  }

  @Override
  public void confer() {
    System.out.println("ProxyStar.confer()");
  }

  @Override
  public void signContract() {
    System.out.println("ProxyStar.signContract()");
  }

  @Override
  public void sing() {
    star.sing();
  }

}

4.測試類別


public class Client {
  public static void main(String[] args) {
    //定义真实对象角色
    Star realStar = new RealStar();
    //定义代理对象角色,内部含有真实对象的引用
    Star proxyStar = new ProxyStar(realStar);

    proxyStar.confer();
    proxyStar.signContract();
    proxyStar.bookTicket();
    proxyStar.sing();
    proxyStar.collectMoney();

  }
}

執行結果如下:


ProxyStar.confer()
ProxyStar.signContract()
ProxyStar.bookTicket()
RealStar.sing()
ProxyStar.collectMoney()

  從上面的例子可以看出代理對象將客戶端的呼叫委派給真實對象,在呼叫目標對象的方法之前跟之後都可以執行特定的操作。

動態代理程式:(動態產生代理類別):

#動態代理程式比起靜態代理程式的優點:

  抽象角色中(介面)聲明的所有方法都被轉移到呼叫伺服器的一個集中的方法中處理,這樣我們可以更加靈活和統一的處理眾多的方法。

JDK自帶的動態代理

#java.lang.reflect.Proxy
動態產生代理類別和物件

java.lang.reflect.InvocationHandler(處理器介面)
可以透過invoke方法實作對真實角色的代理存取

#每次透過Proxy產生代理類物件時都要指定對應的處理器物件

測試程式碼如下:

1、宣告了代理物件和真實物件的共同接口,定義代理物件和真實物件的公共對外方法。


public interface Star {
  /**
   * 面谈
   */
  void confer();
  /**
   * 签合同
   */
  void signContract();
  /**
   * 订票
   */
  void bookTicket();
  /**
   * 唱歌
   */
  void sing();
  /**
   * 收钱
   */
  void collectMoney();
}

2、定義一個真實物件的類,實作抽象介面所提供的方法。


public class RealStar implements Star {

  @Override
  public void bookTicket() {
    System.out.println("RealStar.bookTicket()");
  }

  @Override
  public void collectMoney() {
    System.out.println("RealStar.collectMoney()");
  }

  @Override
  public void confer() {
    System.out.println("RealStar.confer()");
  }

  @Override
  public void signContract() {
    System.out.println("RealStar.signContract()");
  }

  @Override
  public void sing() {
    System.out.println("RealStar.sing()");
  }
}

3.定義一個StarHandler類別實作InvocationHandler處理器接口,可以透過invoke方法實現對真實角色的代理訪問,也可在invoke方法中對眾多操作做統一的處理。


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class StarHandler implements InvocationHandler{

  private Star realStar;

  public StarHandler(Star realStar) {
    super();
    this.realStar = realStar;
  }

  @Override
  public Object invoke(Object proxy, Method method, Object[] args)
      throws Throwable {
    //返回值
    Object object = null;

    System.out.println("真正的方法执行前!");
    System.out.println("面谈,签合同,预付款,订机票");

    if(method.getName().equals("sing")){
      object = method.invoke(realStar, args);
    }

    System.out.println("真正的方法执行后!");
    System.out.println("收尾款");
    return object;

  }

}

4.客户端测试类


import java.lang.reflect.Proxy;

public class Client {

public static void main(String[] args) {

    Star realStar = new RealStar();
    StarHandler handler = new StarHandler(realStar);

    //通过Proxy生成代理类对象并指定对应的处理器对象
    Star proxyStar = (Star)Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(), 
        new Class[]{Star.class}, handler);

    proxyStar.sing();

  }

运行结果如下:

真正的方法执行前!
面谈,签合同,预付款,订机票
RealStar.sing()
真正的方法执行后!
收尾款

开发框架中的应用场景

代理模式在开发框架中的应用场景是非常多的,实际上随便选择一个开发框架都有用到代理模式。例如:

mybatis中实现拦截器插件
AspectJ的实现
spring中AOP的实现

以上是Java設計模式之代理模式的筆記的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn