C#設計模式學習筆記:(16)觀察者模式

来源:https://www.cnblogs.com/atomy/archive/2020/02/20/12334934.html
-Advertisement-
Play Games

本筆記摘抄自:https://www.cnblogs.com/PatrickLiu/p/7928521.html,記錄一下學習過程以備後續查用。 一、引言 今天我們要講行為型設計模式的第四個模式--觀察者模式,先從名字上來看。觀察者模式可以理解為既然有“觀察者”,那肯定就有“被觀察者”了。“觀察者” ...


    本筆記摘抄自:https://www.cnblogs.com/PatrickLiu/p/7928521.html,記錄一下學習過程以備後續查用。

    一、引言

    今天我們要講行為型設計模式的第四個模式--觀察者模式,先從名字上來看。觀察者模式可以理解為既然有“觀察者”,那肯定就有“被觀察者”了。“觀察者”

監視著“被觀察者”,如果“被觀察者”有所行動,“觀察者”就會做出相應的動作來回應。聽起來是不是有點像“諜戰”的味道?比如“諜影重重”那類優秀的影片。

    觀察者模式在現實生活中,實例其實是很多的,比如:八九十年代我們訂閱的報紙,我們會定期收到報紙,因為我們訂閱了。銀行可以給儲戶發手機簡訊,

也是觀察者模式很好的使用的例子,因為我們訂閱了銀行的簡訊業務,當我們賬戶餘額發生變化就會收到通知。還有很多,我就不一一列舉了,發揮大家的

想象吧。好了,接下來,就讓我們看看該模式具體是怎麼實現的吧。

    二、觀察者模式介紹

    觀察者模式:英文名稱--Observer Pattern;分類--行為型。

    2.1、動機(Motivate)

    在軟體構建過程中,我們需要為某些對象建立一種“通知依賴關係”--一個對象(目標對象)的狀態發生改變,所有的依賴對象(觀察者對象)都將得到通知。

如果這樣的依賴關係過於緊密,將使軟體不能很好地抵禦變化。使用面向對象技術,可以將這種依賴關係弱化,並形成一種穩定的依賴關係,從而實現軟體體

繫結構的松耦合。

    2.2、意圖(Intent)

    定義對象間的一種一對多的依賴關係,以便當一個對象的狀態發生改變時,所有依賴於它的對象都得到通知並自動更新。——《設計模式》GoF

    2.3、結構圖

    2.4、模式的組成

    可以看出,在觀察者模式的結構圖有以下角色:

    1)抽象主題角色(Subject):抽象主題把所有觀察者對象的引用保存在一個列表中,並提供增加和刪除觀察者對象的操作,抽象主題角色又叫做抽象被觀

察者角色,一般由抽象類或介面實現。

    2)抽象觀察者角色(Observer):為所有具體觀察者定義一個介面,在得到主題通知時更新自己,一般由抽象類或介面實現。

    3)具體主題角色(ConcreteSubject):實現抽象主題介面,具體主題角色又叫做具體被觀察者角色。

    4)具體觀察者角色(ConcreteObserver):實現抽象觀察者角色所要求的介面,以便使自身狀態與主題的狀態相協調。

    2.5、觀察者模式的具體實現

    觀察者模式在顯示生活中也有類似的例子,比如:我們訂閱銀行簡訊業務,當我們賬戶發生改變,我們就會收到相應的簡訊。類似的還有微信訂閱號,今天

我們以訂閱銀行簡訊業務為例來講講觀察者模式的實現,實現代碼如下:

    class Program
    {
        /// <summary>
        /// 銀行簡訊系統抽象介面,是被觀察者--該類型相當於抽象主題角色Subject。
        /// </summary>
        public abstract class BankMessageSystem
        {
            protected IList<Depositor> observers;

            //構造函數初始化觀察者列表實例
            protected BankMessageSystem()
            {
                observers = new List<Depositor>();
            }

            //增加預約儲戶
            public abstract void Add(Depositor depositor);

            //刪除預約儲戶
            public abstract void Delete(Depositor depositor);

            //通知儲戶
            public void Notify()
            {
                foreach (Depositor depositor in observers)
                {
                    if (depositor.AccountIsChanged)
                    {
                        depositor.Update(depositor.Balance, depositor.OperationDateTime);
                        //賬戶發生變化並且通知後,則可認為賬戶已無變化。
                        depositor.AccountIsChanged = false;
                    }
                }
            }
        }

        /// <summary>
        /// 廣東銀行簡訊系統,是被觀察者--該類型相當於具體主題角色ConcreteSubject。
        /// </summary>
        public sealed class GuangDongBankMessageSystem : BankMessageSystem
        {
            //增加預約儲戶
            public override void Add(Depositor depositor)
            {
                //應該先判斷該用戶是否已預約?如未預約則增加到列表中,這裡簡化了。
                observers.Add(depositor);
            }

            //刪除預約儲戶
            public override void Delete(Depositor depositor)
            {
                //應該先判斷該用戶是否有預約?如有則刪除,否則不操作,這裡簡化了。
                observers.Remove(depositor);
            }
        }

        /// <summary>
        /// 儲戶的抽象介面--相當於抽象觀察者角色(Observer)
        /// </summary>
        public abstract class Depositor
        {
            //儲戶的名稱,假設是唯一的。
            public string Name { get; private set; }

            //儲戶的餘額
            public int Balance { get; private set; }

            //賬戶操作時間
            public DateTime OperationDateTime { get; set; }

            //賬戶是否發生變化
            public bool AccountIsChanged { get; set; }

            //初始化狀態數據
            protected Depositor(string name, int total)
            {
                Name = name;
                Balance = total;            //存款總額等於餘額
                AccountIsChanged = false;   //賬戶未發生變化
            }

            //取錢
            public void GetMoney(int num)
            {
                if (num <= Balance && num > 0)
                {
                    Balance -= num;
                    AccountIsChanged = true;
                    OperationDateTime = DateTime.Now;
                }
            }

            //更新儲戶狀態
            public abstract void Update(int currentBalance, DateTime dateTime);
        }

        /// <summary>
        /// 廣東的具體儲戶--相當於具體觀察者角色ConcreteObserver
        /// </summary>
        public sealed class GuangDongDepositor : Depositor
        {
            public GuangDongDepositor(string name, int total) : base(name, total) { }

            public override void Update(int currentBalance, DateTime dateTime)
            {
                Console.WriteLine(string.Format(Name + ",您的賬戶餘額有變化,發生時間:{0},當前餘額:{1}元。", dateTime.ToString(), currentBalance.ToString()));
            }
        }

        static void Main(string[] args)
        {
            #region 觀察者模式
            //假設有3位儲戶,都是武林高手,也比較有錢。
            Depositor huangFeiHong = new GuangDongDepositor("黃飛鴻", 3000);
            Depositor fangShiYu = new GuangDongDepositor("方世玉", 1300);
            Depositor hongXiGuan = new GuangDongDepositor("洪熙官", 2500);
            BankMessageSystem guangDongBank = new GuangDongBankMessageSystem();

            //這三位開始訂閱銀行簡訊業務
            guangDongBank.Add(huangFeiHong);
            guangDongBank.Add(fangShiYu);
            guangDongBank.Add(hongXiGuan);

            //早上黃飛鴻取了100塊錢
            huangFeiHong.GetMoney(100);
            guangDongBank.Notify();

            //中午黃飛鴻和方世玉各取了200塊
            huangFeiHong.GetMoney(200);
            fangShiYu.GetMoney(200);
            guangDongBank.Notify();

            //晚上他們三個都取了錢
            huangFeiHong.GetMoney(300);
            fangShiYu.GetMoney(300);
            hongXiGuan.GetMoney(300);
            guangDongBank.Notify();

            Console.Read();
            #endregion
        }
    }
View Code

    運行結果如下:

    觀察者模式有些麻煩的地方就是關於狀態的處理,大家可以細細體會一下。模式還是要多寫多練習,裡面的道理就不難理解了。

    三、觀察者模式的實現要點

    使用面向對象的抽象,Observer模式使得我們可以獨立地改變目標與觀察者(面向對象中的改變不是指改代碼,而是指擴展、子類化、實現介面),從而使

二者之間的依賴關係達到松耦合。目標發送通知時,無需指定觀察者,通知(可以攜帶通知信息作為參數)會自動傳播。觀察者自己決定是否需要訂閱通知,

目標對象對此一無所知。

    在C#的event中,委托充當了抽象的Observer介面,而提供事件的對象充當了目標對象。委托是比抽象Observer介面更為松耦合的設計。

    3.1、觀察者模式的優點

    1)觀察者模式實現了表示層和數據邏輯層的分離,並定義了穩定的更新消息傳遞機制,同時抽象了更新介面,使得可以有各種各樣不同的表示層,即觀察者。

    2)觀察者模式在被觀察者和觀察者之間建立了一個抽象的耦合,被觀察者並不知道任何一個具體的觀察者,只是保存著抽象觀察者的列表,每個具體觀察者

都符合一個抽象觀察者的介面。

    3)觀察者模式支持廣播通信。被觀察者會向所有的註冊過的觀察者發出通知。

    3.2、觀察者模式的缺點

    1)如果一個被觀察者有很多直接和間接的觀察者時,將所有的觀察者都通知到會花費很多時間。

    2)雖然觀察者模式可以隨時使觀察者知道所觀察的對象發送了變化,但是觀察者模式沒有相應的機制使觀察者知道所觀察的對象是怎樣發生變化的。

    3)如果在被觀察者之間有迴圈依賴的話,被觀察者會觸發它們之間進行迴圈調用,導致系統崩潰,在使用觀察者模式應特別註意這點。

    四、.NET中觀察者模式的實現

    其實在Net裡面實現的觀察者模式做了一些改變,用委托或者說是事件來實現觀察者模式。事件我們都很明白,可以註冊控制項的事件,當觸發控制項的動作時候,

相應的事件就會執行,在事件的執行過程中我們就可以做相關的提醒業務。這裡關於觀察者模式在Net裡面的實現就不說了,如果大家不明白,可以多看看相關委

托或者事件的相關資料。

    五、總結

    這個模式結合實例理解是很容易的,模式的使用我們不能照搬,要理解,當然多多聯繫和寫代碼也是必不可少的,我們使用模式的一貫宗旨是通過重構和迭代,

在我們的代碼中實現相應的模式。


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

-Advertisement-
Play Games
更多相關文章
  • 0、前言 用mybatis,那麼分頁必不可少,基本都是用PageHelper這個分頁插件,好用方便; 1、實現 1.1、添加依賴: <!-- 3、集成 mybatis pagehelper--> <dependency> <groupId>com.github.pagehelper</groupId ...
  • 多用www.bing.com國際版解決代碼報錯 代碼運行的時候,報異常,國內的搜索引擎一搜, 浮誇的廣告太多,解決方案准確性不足, 盜版又很嚴重(導致一錯皆錯),方案未及時更新等詬病。 www.bing.com國際版可以關聯到: (1). 外國官網,可以獲得官方的解決方案。 (2). stackov ...
  • 擱置了幾天,工作忙的一塌糊塗,今天終於抽空來繼續看看MVC的知識。先來看看MVC的路由是如何處理的。以下為替代的路由: app.UseEndpoints(endpoints => { endpoints.MapControllerRoute( name: "default", pattern: "{ ...
  • 前言 通常在應用程式開發到正式上線,在這個過程中我們會分為多個階段,通常會有 開發、測試、以及正式環境等。每個環境的參數配置我們會使用不同的參數,因此呢,在ASP.NET Core中就提供了相關的環境API,方便我們更好的去做這些事情。 環境 ASP.NET Core使用ASPNETCORE_ENV ...
  • ​ 在C#8.0中,針對介面引入了一項新特性,就是可以指定預設實現,方便對已有實現進行擴展,也對面向Android和Swift的Api進行互操作提供了可能性。下麵我們來看看該特性的具體規則與實現。 一、主要應用場景: 在不破壞影響已有實現的情況下,可以添加新成員。這解決了在第三方已經大量使用了的介面 ...
  • 一文帶你瞭解 C DLR 的世界 在很久之前,我寫了一片文章 "dynamic結合匿名類型 匿名對象傳參" ,裡面我以為DLR內部是用反射實現的。因為那時候是心中想當然的認為只有反射能夠在運行時解析對象的成員信息並調用成員方法。後來也是因為其他的事一直都沒有回過頭來把這一節知識給補上,正所謂亡羊補牢 ...
  • 首先新建一個項目,名稱叫Caliburn.Micro.ActionConvertions 然後刪掉MainWindow.xaml 然後去app.xaml刪掉StartupUri這行代碼 其次,安裝Caliburn.Micro,Caliburn.Micro.Core,這兩個Nuget包,如下圖 然後新 ...
  • 先看核心代碼: public List<DataEntity> SearchShopSalesReport(DateTimeOffset? dateFrom, DateTimeOffset? dateTo,string groupBy) { var query = data.DataEntity / ...
一周排行
    -Advertisement-
    Play Games
  • Timer是什麼 Timer 是一種用於創建定期粒度行為的機制。 與標準的 .NET System.Threading.Timer 類相似,Orleans 的 Timer 允許在一段時間後執行特定的操作,或者在特定的時間間隔內重覆執行操作。 它在分散式系統中具有重要作用,特別是在處理需要周期性執行的 ...
  • 前言 相信很多做WPF開發的小伙伴都遇到過表格類的需求,雖然現有的Grid控制項也能實現,但是使用起來的體驗感並不好,比如要實現一個Excel中的表格效果,估計你能想到的第一個方法就是套Border控制項,用這種方法你需要控制每個Border的邊框,並且在一堆Bordr中找到Grid.Row,Grid. ...
  • .NET C#程式啟動閃退,目錄導致的問題 這是第2次踩這個坑了,很小的編程細節,容易忽略,所以寫個博客,分享給大家。 1.第一次坑:是windows 系統把程式運行成服務,找不到配置文件,原因是以服務運行它的工作目錄是在C:\Windows\System32 2.本次坑:WPF桌面程式通過註冊表設 ...
  • 在分散式系統中,數據的持久化是至關重要的一環。 Orleans 7 引入了強大的持久化功能,使得在分散式環境下管理數據變得更加輕鬆和可靠。 本文將介紹什麼是 Orleans 7 的持久化,如何設置它以及相應的代碼示例。 什麼是 Orleans 7 的持久化? Orleans 7 的持久化是指將 Or ...
  • 前言 .NET Feature Management 是一個用於管理應用程式功能的庫,它可以幫助開發人員在應用程式中輕鬆地添加、移除和管理功能。使用 Feature Management,開發人員可以根據不同用戶、環境或其他條件來動態地控制應用程式中的功能。這使得開發人員可以更靈活地管理應用程式的功 ...
  • 在 WPF 應用程式中,拖放操作是實現用戶交互的重要組成部分。通過拖放操作,用戶可以輕鬆地將數據從一個位置移動到另一個位置,或者將控制項從一個容器移動到另一個容器。然而,WPF 中預設的拖放操作可能並不是那麼好用。為瞭解決這個問題,我們可以自定義一個 Panel 來實現更簡單的拖拽操作。 自定義 Pa ...
  • 在實際使用中,由於涉及到不同編程語言之間互相調用,導致C++ 中的OpenCV與C#中的OpenCvSharp 圖像數據在不同編程語言之間難以有效傳遞。在本文中我們將結合OpenCvSharp源碼實現原理,探究兩種數據之間的通信方式。 ...
  • 一、前言 這是一篇搭建許可權管理系統的系列文章。 隨著網路的發展,信息安全對應任何企業來說都越發的重要,而本系列文章將和大家一起一步一步搭建一個全新的許可權管理系統。 說明:由於搭建一個全新的項目過於繁瑣,所有作者將挑選核心代碼和核心思路進行分享。 二、技術選擇 三、開始設計 1、自主搭建vue前端和. ...
  • Csharper中的表達式樹 這節課來瞭解一下表示式樹是什麼? 在C#中,表達式樹是一種數據結構,它可以表示一些代碼塊,如Lambda表達式或查詢表達式。表達式樹使你能夠查看和操作數據,就像你可以查看和操作代碼一樣。它們通常用於創建動態查詢和解析表達式。 一、認識表達式樹 為什麼要這樣說?它和委托有 ...
  • 在使用Django等框架來操作MySQL時,實際上底層還是通過Python來操作的,首先需要安裝一個驅動程式,在Python3中,驅動程式有多種選擇,比如有pymysql以及mysqlclient等。使用pip命令安裝mysqlclient失敗應如何解決? 安裝的python版本說明 機器同時安裝了 ...