設計模式七大原則—開閉原則

来源:https://www.cnblogs.com/green-jcx/archive/2022/05/07/16241824.html
-Advertisement-
Play Games

重要性 有過一些實際開發工作的朋友一定對某個場景會深有體會,那就是客戶經常會對現有的功能提出新的需求要我們改動,並且要快速完成。如果你的代碼沒有很好的遵循“開閉原則”,並且頂著工期的縮減,那我們對需求變化的修改,“往往就像在一個草稿紙上反覆的塗抹”,隨著不斷的變化修改代碼就會顯得很亂,可能到最後你連 ...


重要性

有過一些實際開發工作的朋友一定對某個場景會深有體會,那就是客戶經常會對現有的功能提出新的需求要我們改動,並且要快速完成。如果你的代碼沒有很好的遵循“開閉原則”,並且頂著工期的縮減,那我們對需求變化的修改,“往往就像在一個草稿紙上反覆的塗抹”,隨著不斷的變化修改代碼就會顯得很亂,可能到最後你連自己的代碼都看不懂了,還可能影響現有的功能(“賠了夫人又折兵”)

 定義

開閉原則在定義描述上其實非常的簡短,那就是:“對擴展開放,對修改關閉”,該原則是編程種最基本、最重要的設計原則。其實在經過實際的開發工作後,大家都自然而然的會體會到這個開閉原則的思想:就是我們在對現有功能進行調整修改的時候,我們的調整修改儘量做到,對現有的功能不受影響,這樣才能保障我們系統的穩定性。

如果要做到這一點,我們的程式設計就必須遵循開閉原則,即軟體模塊在變動的情況下,是通過新增擴展單獨的代碼來實現變動,而不是去修改現有的代碼來完成變動。開閉原則並不是特指某種技術,他是建立在面向對象編程的運用基礎之上的一種思想,所以沒有固定的套路,要實現該原則,那麼你就要保障你的程式能夠“對擴展開放,對修改關閉”即可。

打個比方

“對擴展開放,對修改關閉”,我們打個比方來理解以下這個較為生硬的概念。比方說你現在有臺電腦它的磁碟空間不夠,因此此時如果要增加現有的磁碟空間,目前的電腦就需要把機箱拆掉將磁碟返廠處理,如果你要求的新增空間特別大,很可能還需要換主板。那麼對於這種情況,實際上它就沒有做到對“修改關閉”,因為我們的變動需要改變現有的內容。

如果說電腦在設計之初就考慮到了可能發生的變化,因此設計了外部的擴展介面,那麼我們就只需要買一個新的硬碟,插入到擴展介面就可以完成磁碟空間的增加。那麼對於這種情況,它就是滿足了開閉原則的“對擴展開放,對修改關閉”,這個道理在程式設計上同樣如此。

 

代碼示例

下麵通過代碼層面來加深對的理解,通過兩個示例來看看“使用”開閉原則和“未使用”開閉原則的區別,以及如何在代碼層面體現出開閉原則。

1.“未使用”開閉原則

 1    //電腦類
 2     class Computer
 3     {
 4         //讀取設備
 5         public void ReadDevice(int type)
 6         {
 7             if (type==1)
 8             {
 9                 RunMouse();//運行滑鼠
10             }
11             else if (type == 2)
12             {
13                 RunKeyboard();//運行鍵盤
14             }
15             else if (type == 3)
16             {
17                 RunHeadset();   //運行耳機
18             }
19         }
20 
21         //運行滑鼠
22         public void RunMouse()
23         {
24             Console.WriteLine("運行滑鼠");
25         }
26 
27         //運行鍵盤
28         public void RunKeyboard()
29         {
30             Console.WriteLine("運行鍵盤");
31         }
32 
33         //運行耳機
34         public void RunHeadset()
35         {
36             Console.WriteLine("運行耳機");
37         }
38 
39     }
40 
41      
42     internal class Program
43     {
44         static void Main(string[] args)
45         {
46             Computer computer = new Computer();
47             computer.ReadDevice(1);
48         }
49     }

代碼示例簡單也比較貼近實際生活場景,其中主要是通過電腦類中“讀取設備”的方法來讀取不同的設備(滑鼠、鍵盤、耳機),該方法根據傳入的類型參數,來判斷來讀取具體的設備。

對於該示例而言,它實際上就沒有遵循開閉原則,那就是沒有“對擴展開放,對修改關閉”。為什麼這麼說?因為如果現在電腦類需要增加讀取的設備,對於這個示例只能修改現有電腦類的代碼,也就是新增設備運行方法,並加入新的else if判斷類型讀取,並且並不支持單獨的擴展方式對其進行修改。如果無法預料到所有新的讀取設備,那麼你的電腦類將“有無寧日”,代碼可能會像下圖:

 

 2.“使用”開閉原則

 

 1    public abstract class Device
 2     {
 3         public abstract void Run();
 4     }
 5 
 6     public class Mouse : Device
 7     {
 8         public override void Run()
 9         {
10             Console.WriteLine("運行滑鼠");
11         }
12     }
13 
14     public class Keyboard : Device
15     {
16         public override void Run()
17         {
18             Console.WriteLine("運行鍵盤");
19         }
20     }
21     public class Headset : Device
22     {
23         public override void Run()
24         {
25             Console.WriteLine("運行耳機");
26         }
27     }
28 
29 
30     //電腦類
31     class Computer
32     {
33         //讀取設備
34         public void ReadDevice(Device device)
35         {
36             device.Run();
37         }
38 
39     }
40 
41     internal class Program
42     {
43         static void Main(string[] args)
44         {
45             Computer computer = new Computer();
46             computer.ReadDevice(new Headset());
47         }
48     }

該示例將經常變化的部分單獨的抽離了出來,聲明瞭一個抽象設備類,並且根據不同的設備創建繼承該抽象類的實現類。那麼這樣一來就將設備的變化與電腦類隔離開了,後面不管新增還是刪除設備,電腦類都不用進行修改,而新增設備只用創建一個繼承抽象設備類的實現類即可。因此也遵循了開閉原則的“對擴展開放,對修改關閉”。

知識改變命運
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本文收集了各種資料庫的SQL語句優化原理思路、技術要點和方法實操案例文檔!希望大家都能寫得一手好SQL、掌握資料庫高性能運行秘訣! ...
  • HarmonyOS Connect智能硬體開放生態即將步入富設備產業化時代!為了讓廣大開發者能搶先體驗鴻蒙智聯富設備開發,本期我們將為大家帶來七款支持富設備開發的開發板。 ...
  • 精準推送是移動端產品留存階段的主要運營手段,精準推送常常會與用戶畫像緊密結合,針對用戶的喜好、畫像,採用不同策略,但基於用戶所屬區域推送消息卻很難實現。目前市面上大多數第三方消息推送服務商,在系統未深度定製的情況下,通常不支持將推送人群範圍精確到某個商圈或較小的區域,而地理圍欄技術可以很好地彌補這一 ...
  • 前言 本文主要是整理了使用WebRTC做音視頻通訊時的各知識點及問題點。有理解不足和不到位的地方也歡迎指正。 對於你感興趣的部分可以選擇性觀看。 WebRTC的初始化 在使用WebRTC的庫之前,需要對WebRTC進行初始化, 用到的代碼如下: RTCInitializeSSL(); 轉定義後可以看 ...
  • 今天的內容有意思了,朋友們繼續對我們之前的案例完善,是這樣的我們之前是不是靠props來完成父給子,子給父之間傳數據,其實父給子最好的方法就是props但是自給父就不是了,並且今天學下來,不僅如此,組件間任何層級的關係我都可以傳數據了,兄弟之間,爺孫之間等等等等 七.瀏覽器本地存儲 1.localS ...
  • 移動端瀑布流佈局是一種比較流行的網頁佈局方式,視覺上來看就是一種像瀑布一樣垂直落下的排版。每張圖片並不是顯示的正正方方的,而是有的長有的短,呈現出一種不規則的形狀。但是它們的寬度通常都是相同的 因為移動端瀑布流佈局主要為豎向瀑布流,因此本文所探討的是豎向瀑布流 特點 豎向瀑布流佈局主要有下麵幾種特點 ...
  • 可以將( 0, null, false, undefined, NaN )理解為數字 0 與運算: 與運算 類比四則運算中的乘法。0和任何數相乘都等於0,因此他們和其他值做與運算都等於0(等於他本身,例如:null && 'abc',結果為 null;1414 && 0,結果為 0)。 若是兩個0 ...
  • 線程和進程是電腦操作系統的基礎概念,在程式員中屬於高頻辭彙,那如何理解呢?Node.js 中的進程和線程又是怎樣的呢? 一、進程和線程 1.1、專業性文字定義 進程(Process),進程是電腦中的程式關於某數據集合上的一次運行活動,是系統進行資源分配和調度的基本單位,是操作系統結構的基礎,進程 ...
一周排行
    -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中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...