DesignPattern系列__06迪米特原則

来源:https://www.cnblogs.com/JackHou/archive/2019/08/06/11311107.html
-Advertisement-
Play Games

迪米特原則定義 迪米特原則,也叫最少知道原則,即一個類應該 對自己依賴的類知道的越少越好 ,而你被依賴的類多麼複雜,對我都沒有關係。也就是說,對於別依賴的類來說,不管業務邏輯多麼複雜,都應該儘量封裝在類的內部;對外除了必備的public方法,不再泄露任何信息。 1.問題由來 我們知道,類和類是有耦合 ...


迪米特原則定義

迪米特原則,也叫最少知道原則,即一個類應該對自己依賴的類知道的越少越好,而你被依賴的類多麼複雜,對我都沒有關係。也就是說,對於別依賴的類來說,不管業務邏輯多麼複雜,都應該儘量封裝在類的內部;對外除了必備的public方法,不再泄露任何信息。

1.問題由來

我們知道,類和類是有耦合關係的,關係越密切的兩個類之間,其耦合關係越大。

2.對應措施

迪米特原則要求:一個類應該只和之間的直接朋友通信。

1.直接朋友的定義

在上面我們提到了“直接朋友”這個概念,其實,在一個程式對象中,每個類都會和其他類有耦合關係,不然也就沒有必要存在了。耦合的方式包括:依賴、關聯、組合、聚合等。我們說,有耦合關係的兩個類之間的關係,就是朋友關係。
那麼,什麼是“直接朋友”呢?
例如A類和B類具有耦合關係,若A類作為B類的成員變數、方法的形參、返回值,則說這兩個類就是直接朋友;若A類作為B類的方法內的局部變數,則A類就不是B類的直接朋友。也就是說,陌生的類最好不要以局部變數的形式出現在類的內部。

3.應用實踐

迪米特原則要求我們做到以下四點:

1.只和直接朋友溝通

為了說明這點,我們需要一個例子:比如在一所大學內有各個學院,現在要求列印出各個學院和學校總部的員工ID。代碼演示如下:

 public class Demeter1 {
    public static void main(String[] args) {
        SchoolManager schoolManager = new SchoolManager();
        schoolManager.printAllEmp(new CollegeManager());
    }
}

class SchoolManager {
    public void printAllEmp(CollegeManager collegeManager) {
        List<Employee> empList = this.getAllEmployee();
        System.out.println("列印學校總部的員工");
        for (Employee employee: empList) {
            employee.printId();
        }
         //分析問題
    //1. 這裡的 CollegeEmployee 不是  SchoolManager的直接朋友
    //2. CollegeEmployee 是以局部變數方式出現在 SchoolManager
        //3. 違反了 迪米特法則 
        List<CollegeEmployee> collegeEmpList = collegeManager.getAllEmployee();
        System.out.println("列印學院員工");
        for (CollegeEmployee collegeEmployee: collegeEmpList) {
            collegeEmployee.printId();
        }
    }

    //返回所用總部信息
    public List<Employee> getAllEmployee() {
        List<Employee> list = new ArrayList<>();
        //添加5名總部的員工
        for (int i=0; i<5;i++) {
            Employee employee = new Employee();
            employee.setId(i);
            list.add(employee);
        }
        return list;
    }
}

//學院員工的管理類
class CollegeManager {
    //返回學院的所有員工
    public List<CollegeEmployee> getAllEmployee() {
        List<CollegeEmployee> list = new ArrayList<>();
        //添加10名學院員工
        for (int i = 0; i < 10; i++) {
            CollegeEmployee emp = new CollegeEmployee();
            emp.setId(i);
            list.add(emp);
        }
        return list;
    }
}

//學校員工類
class Employee {
    private Integer id;

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public void printId() {
        System.out.println("學校員工,ID=" + this.getId());
    }
}

//學院員工類
class CollegeEmployee {
    private Integer id;

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    public void printId() {
        System.out.println("學院員工,ID=" + this.getId());
    }
}

根據上面的代碼,我們來找一下類SchoolManager的直接朋友:Employee、CollegeManager,而CollegeEmployee 是以局部變數方式出現在 SchoolManager,這就違背了“迪米特原則”。

改進措施

既然如此,我們就要將CollegeEmployee從SchoolManager類中抽離出來,使其不被依賴。
在CollegeManager中增加一個列印學院員工的方法printCollegeEmps(),這樣,SchoolManager就只需調用這個方法就行了。

  public void printCollegeEmps() {
        List<CollegeEmployee> list = this.getAllEmployee();
        for (CollegeEmployee collegeEmployee: list) {
            collegeEmployee.printId();
        }
    }

2.和朋友也要保持適當距離

看到這裡你可能會困惑,既然已經要求我們做到:一個類只和直接朋友溝通,那麼為什麼還要保持距離呢?還是舉例說明:現在有兩個類A、B,類A中有三個public方法,類B需要調用A中的三個方法來完成一個流程:

public class Demeter2 {
    public static void main(String[] args) {
        A a = new A();
        B b = new B();
        b.invokerA(a);
    }
}

class A {
    public void method1() {
        System.out.println("執行第一個方法");
    }

    public void method2() {
        System.out.println("執行第二個方法");
    }

    public void method3() {
        System.out.println("執行第三個方法");
    }

}

class B {
    public void invokerA(A a) {
        System.out.println("調用A的buildMethod()...");
        a.method1();
        a.method2();
        a.method3();
    }
}

OK,功能完成了,但是,類B需要依次調用類A的三個方法,需要保持三個方法對類B都是可見的。也就是說,類B和類A的耦合度太高了,我們可以改善一下兩者的關係,適度的降低一下耦合度。在類A中定義一個public方法,來封裝類B的邏輯,其餘方法設置為private。

 //類A的相應修改
 private void method1() {
        System.out.println("執行第一個方法");
    }

    private void method2() {
        System.out.println("執行第二個方法");
    }

    private void method3() {
        System.out.println("執行第三個方法");
    }

    public void buildMethod() {
        System.out.println("流程開始");
        method1();
        method2();
        method3();
        System.out.println("流程結束");
    }

3.是自己的就是自己的

當一個方法放在本類和其他類中都可以的時候,那麼,如果,一個方法放在本類中,不會增加類之間的耦合度,也不會造成不良影響,放在本類中

4.慎用Serializable

舉例來說,在一個項目中使用RMI方式傳遞一個VO對象時,這個對象就必須實現Serializable介面,也就是進行序列化。當你突然將客戶端這個VO對象的訪問許可權從private更改為public的時候,而服務端沒有進行對應的改變,就會出現錯誤。

4.迪米特原則的註意事項和細節

1.迪米特原則的核心就是降低類之間的耦合。只有耦合降低了,類的復用率才能提高。

2.註意事項:

凡事講究適度,迪米特原則要求降低類之間的耦合,並不是要求沒有耦合。


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

-Advertisement-
Play Games
更多相關文章
  • 1. filter針對數組起過濾作用篩選出符合條件的一個或多個元素 lvar newarr = [ { num: 1, val: 'ceshi', flag: 'aa' }, { num: 2, val: 'ww' } ] console.log(newarr.filter(item => item ...
  • 1. typeof 運算符 typeof 可以判斷基本數據類型: typeof 123; // "number" typeof 'abc'; // "string" typeof true; // "boolean" 碰到複合數據類型的情況: typeof {}; // "object" typeo ...
  • 1.1什麼是函數提升和變數的提升? JS引擎在運行整個JS代碼的過程中,分為倆步。 第一步是讀取和解析JS代碼,第二部是執行。 在引擎解析JS代碼的時候,當解析器遇見變數聲明(var 變數名)和函數聲明 (function 函數名)的時候,會將這些聲明提到各自作用域的最前面。 1.2 作用域 在ES ...
  • 1、構建項目 上面的第一條,也就是 aaa 這一個選項在你第一次創建項目的時候是並不會出現的,只有你第一次創建完成項目後回提示你保存為預設配置模板,下次新建項目的時候就可以使用你選用的配置快速新建項目了,不需要再重新選擇配置項目了。 第二條選項便是 vue cli 3 預設的項目模板,包含 babe ...
  • 在前一篇文章中,我們介紹瞭如何在JavaScript中實現集合。字典和集合的主要區別就在於,集合中數據是以[值,值]的形式保存的,我們只關心值本身;而在字典和散列表中數據是以[鍵,值]的形式保存的,鍵不能重覆,我們不僅關心鍵,也關心鍵所對應的值。 我們也可以把字典稱之為映射表。由於字典和集合很相似, ...
  • jQuery 滑動方法有三種:slideDown()、slideUp()、slideToggle()。 ...
  • TypeScript學習隨筆(一) 這麼久了還不沒好好學習哈這麼火的ts,邊學邊練邊記吧! 啥子是TypeScript TypeScript 是 JavaScript 的一個超集,支持 es6 標準。 TypeScript 由微軟開發的自由和開源的編程語言。 TypeScript 設計目標是開發大型 ...
  • 前言 現在,很少有人和90年代一樣,自己去實現一個軟體的各個方面,也就是說,在工作中,和人溝通是必備的技能。那麼,作為一枚碼農,如何和他人溝通呢?這就要依靠本文的主題了——UML。 簡介 UML——Unified modeling language UML(統一建模語言),是一種用於軟體系統分析和設 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...