java設計模式6——代理模式

来源:https://www.cnblogs.com/xgp123/archive/2020/02/16/12308415.html
-Advertisement-
Play Games

java設計模式6——代理模式 1、代理模式介紹: 1.1、為什麼要學習代理模式?因為這就是Spring Aop的底層!(SpringAop 和 SpringMvc) 1.2、代理模式的分類: 靜態代理 動態代理 1.3、代理模式關係圖(以租房子為例) 2、靜態代理 2.1、角色分析: 抽象角色:一 ...


java設計模式6——代理模式

1、代理模式介紹:

1.1、為什麼要學習代理模式?因為這就是Spring Aop的底層!(SpringAop 和 SpringMvc)

1.2、代理模式的分類:

  • 靜態代理
  • 動態代理

1.3、代理模式關係圖(以租房子為例)

2、靜態代理

2.1、角色分析:

  1. 抽象角色:一般會使用介面或者抽象類來解決
  2. 真實角色:被代理的角色
  3. 代理客戶:代理真實角色。代理真實角色後,我們一般會做一些附屬的操作
  4. 客戶:訪問代理對象的人

2.2、例1(租房子演示)

2.2.1、抽象角色實現(房東抽象類)

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo1;

/**
 * 租房
 */
public interface Rent {

    void rent();
}

2.2.2、真實角色實現(房東的實現類)

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo1;

/**
 * 房東
 */
public class Host implements Rent {

    @Override
    public void rent() {
        System.out.println("房東要出租房子!");
    }
}

2.2.3、不通過代理進行租房

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo1;

public class Client {
    public static void main(String[] args) {
        //直接找房東
        Host host = new Host();
        host.rent();
    }
}

運行結果:

房東要出租房子!

2.2.4、弊端

  1. 在實際的生活場景中,房東可能只賣房,而不負責租房
  2. 如果要提高租房子的成功率,必然還需要一些其他的服務,如:看房、談價錢、簽合同等,而這些房東並不想參與,因此產生了中介,也就是代理。

2.2.5、代理出現

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo1;

public class Proxy implements Rent {

    private Host host;

    public Proxy() {

    }

    public Proxy(Host host) {
        this.host = host;
    }

    @Override
    public void rent() {
        seeHouse();
        host.rent();
        fare();
        hetong();
    }

    //看房
    public void seeHouse() {
        System.out.println("中介代理看房");
    }

    //收中介費
    public void fare() {
        System.out.println("收中介費");
    }

    public void hetong() {
        System.out.println("簽合同");
    }
}

2.2.6、通過代理來租房子

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo1;

public class Client {
    public static void main(String[] args) {
/*        //直接找房東
        Host host = new Host();
        host.rent();*/

        //通過代理去租房子
        //房東要租房子
        Host host = new Host();
        //代理
        //中介幫房東租房子,但是,中介一半都會有一些附屬操作
        Proxy proxy = new Proxy(host);
        proxy.rent();
    }
}

運行結果:

中介代理看房
房東要出租房子!
收中介費
簽合同

2.3、分析:

代理模式的好處

  1. 可以使真實角色的操作更加純粹,不用去關註一些公共的業務。
  2. 公共的業務交給了代理角色去完成,實現了業務的分工。
  3. 公共業務發生擴展的時候,方便集中管理。

缺點:

  • 一個真實角色就會產生一個代理的角色,代碼量翻倍,開發效率變低。

2.4、例1(增刪改查業務的演示)

2.4.1、編寫service的抽象類

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo2;

public interface UserService {
    void add();
    void del();
    void update();
    void query();
}

2.4.2、編寫service的實現類

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo2;

/**
 * 真實對象
 */
public class UserServiceImpl implements UserService {
    @Override
    public void add() {
        System.out.println("增加了一個用戶");
    }

    @Override
    public void del() {
        System.out.println("刪除了一個用戶");
    }

    @Override
    public void update() {
        System.out.println("更新了一個用戶");
    }

    @Override
    public void query() {
        System.out.println("查詢了一個用戶");
    }
}

2.4.3、如果此時想不改變原有的代碼,而增加方法日誌列印的功能,此時代理誕生

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo2;

/**
 * 代理角色,增加日誌的功能
 */
public class UserServiceProxy implements UserService {

    private UserService userService;

    public void setUserService(UserService userService) {
        this.userService = userService;
    }

    @Override
    public void add() {
        userService.add();
        log("add");
    }

    @Override
    public void del() {
        userService.del();
        log("del");
    }

    @Override
    public void update() {
        userService.update();
        log("update");
    }

    @Override
    public void query() {
        userService.query();
        log("query");
    }

    public void log(String msg) {
        System.out.println("使用了" + msg + "方法!");
    }
}

2.4.4、編寫客戶端類,使用service的方法,並列印日誌

package com.xgp.company.結構性模式.代理模式.靜態代理.Demo2;

public class Client {
    public static void main(String[] args) {
        UserService userService = new UserServiceImpl();

        UserServiceProxy proxy = new UserServiceProxy();
        proxy.setUserService(userService);

        proxy.add();
        proxy.del();
        proxy.update();
        proxy.query();
    }
}

運行結果:

增加了一個用戶
使用了add方法!
刪除了一個用戶
使用了del方法!
更新了一個用戶
使用了update方法!
查詢了一個用戶
使用了query方法!

2.5、反思:在實際的業務中,橫向開發的好處

可以保證原有代碼的可行性,不會對軟體原來的版本進行破壞,向過去相容。

3、動態代理

3.1、動態代理的特點

  1. 動態代理和靜態代理一樣
  2. 動態代理的代理類是動態生成的,不是我們直接寫的

3.2、動態代理實現方式的分類

  • 基於介面——JDK動態代理(本節需要講述的方式)
  • 基於類:cglib
  • java位元組碼實現:javasist

3.3、需要瞭解的兩個類

  • Proxy:代理類
  • InvocationHandler:調用處理程式類

具體的使用參照java的api

3.4、例1——房東租房的例子

3.4.1、房東的介面和實現類不變

3.4.2、實現自動生成代理的類

package com.xgp.company.結構性模式.代理模式.動態代理.Demo1;

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

/**
 * 等下我們會用這個類,自動生成代理類
 */
public class ProxyInvocationHandler implements InvocationHandler {

    /**
     * 被代理的介面
     */
    private Rent rent;

    public void setRent(Rent rent) {
        this.rent = rent;
    }

    /**
     * 獲得一個代理
     * @return
     */
    public Object getProxy() {
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),rent.getClass().getInterfaces(),this);
    }

    /**
     * 處理代理實例,並放回結果
     * @param proxy
     * @param method
     * @param args
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //用反射來執行方法
        seeHouse();     //調用代理自身的方法
        //動態代理的本質,就是使用反射機制來實現
        Object result = method.invoke(rent, args);
        fare();
        hetong();
        return result;
    }

    public void seeHouse() {
        System.out.println("中介帶看房子!");
    }

    //收中介費
    public void fare() {
        System.out.println("收中介費");
    }

    public void hetong() {
        System.out.println("簽合同");
    }
}

3.4.3、實現客戶端使用代理來租房子

package com.xgp.company.結構性模式.代理模式.動態代理.Demo1;

public class Client {
    public static void main(String[] args) {
        //真實角色
        Host host = new Host();

        //代理角色
        ProxyInvocationHandler pih = new ProxyInvocationHandler();
        //通過調用程式處理角色,來處理我們要調用的介面對象
        pih.setRent(host);

        //放回代理類
        //這裡的代理類就是動態生成的,我們並沒有寫他
        Rent proxy = (Rent) pih.getProxy();
        proxy.rent();
    }
}

運行結果:

中介帶看房子!
房東要出租房子!
收中介費
簽合同

3.5、例2——增刪改查的例子

3.5.1、先來實現一個萬能的代理生成類

package com.xgp.company.結構性模式.代理模式.動態代理.Demo2;

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

/**
 * 萬能代理類
 */
public class ProxyInvocationHandler implements InvocationHandler {

    /**
     * 被代理的介面
     */
    private Object target;

    public void setTarget(Object target) {
        this.target = target;
    }

    /**
     * 獲得一個代理
     * @return
     */
    public Object getProxy() {
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
    }

    /**
     * 處理代理實例,並放回結果
     * @param proxy
     * @param method
     * @param args
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //用反射來執行方法
        //動態代理的本質,就是使用反射機制來實現
        Object result = method.invoke(target, args);
        log(method.getName());
        return result;
    }

    public void log(String msg) {
        System.out.println("使用了" + msg + "方法!");
    }
}

也就是將上一個例子的具體實現類——房東類,替換成Object類

3.5.2、UserService的介面和實現類不變

3.5.3、編寫客戶端類使用代理調用增刪改查方法

package com.xgp.company.結構性模式.代理模式.動態代理.Demo2;

public class Client {
    public static void main(String[] args) {
        //真實角色
        UserService userService = new UserServiceImpl();
        //代理角色
        ProxyInvocationHandler pih = new ProxyInvocationHandler();
        pih.setTarget(userService);
        UserService proxy = (UserService) pih.getProxy();

        proxy.add();
        proxy.del();
        proxy.update();
        proxy.query();
    }
}

運行結果:

增加了一個用戶
使用了add方法!
刪除了一個用戶
使用了del方法!
更新了一個用戶
使用了update方法!
查詢了一個用戶
使用了query方法!

3.6、弊端分析:

雖然使用動態代理能夠節省代碼量,並且實現靜態代理的全部優點。但是,動態代理的核心是反射技術,通過反射技術調用方法效率較大,因此也可能影響系統效率。

3.7、動態代理的好處

一個動態代理的是一個介面,一般就是對應的一類業務。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 由於程式都是自上向下的順序執行的,所以通過流程式控制制語句可以改變程式執行的順序,或者反覆的執行某一段的程式。 語句的分類 1. 條件判斷語句 2. 條件分支語句 3. 迴圈語句 條件判斷語句 條件判斷語句也稱為if語句 語法一: if(條件表達式){ 語句... } 執行流程:if語句執行時,會先對條 ...
  • java的基本數據類型一共有 byte short int long float double char boolean js中定義變數使用關鍵字 var js的原始類型(五個) String: 字元串 在字元串中我們可以使用\作為轉移符號。比如 \" ",\' ',\\ \,\n 表示換行 Num ...
  • 選擇器是jquery的核心 jquery選擇器返回的對象是jquery對象,不會返回undefined或者null,因此不必進行判斷 基本選擇器: ID選擇器 $("#ID") element選擇器(標簽選擇器)$("標簽") 如 $("div") 獲取具體的標簽,使用數組下標的方式 $("div" ...
  • jquery是一個快速、小巧,功能強大的javascript函數庫。 jquery主要用來替代原生的javascript,簡化代碼。 前端最頭疼的就是相容:IE6/7/8相容的最高版本是jQuery1.9.1 jQuery的操作在一個類中,不會污染頂級變數 基本不用考慮瀏覽器的相容性 支持鏈式操作方 ...
  • JS中將數組轉換為鏈表 javascript / 將數組轉換為鏈表 @param array arr 需要轉換的數組 @param int type 轉換的類型,0為單鏈表,1為迴圈鏈表 @return object 返回鏈表 / function array2List(arr, type = 0) ...
  • 面試題:如何用apply實現一個bind? ...
  • 1.問題描述: 在日常練習的時候遇見的錯誤,雖然頁面可以順利顯示,但後臺報錯:(雖然不影響導出效果,但看到後臺的異常,內心還是不舒服的) java.lang.IllegalStateException: getOutputStream() has already been called for th ...
  • 切記:用table標簽來佈局form表單元素,table標簽必須放在form表單內部,否則可能會出現各種bug 原文地址:https://blog.csdn.net/weixin_43343144/article/details/88847275 表單與表格的嵌套必須將表單寫在外面,因為表格有嵌套規 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...