.net core 常見設計模式-IChangeToken

来源:https://www.cnblogs.com/jionsoft/archive/2020/02/01/12249326.html
-Advertisement-
Play Games

場景 一個對象A,希望它的某些狀態在發生改變時通知到B(或C、D),常見的做法是在A中定義一個事件(或直接用委托),當狀態改變時A去觸發這個事件。而B直接訂閱這個事件 這種設計有點問題B由於要訂閱A的事件,所以B得完全引用A,其實有時候沒必要,因為我只關心A的狀態變化而已狀態變更通知這種場景很多,有 ...


場景

一個對象A,希望它的某些狀態在發生改變時通知到B(或C、D),
常見的做法是在A中定義一個事件(或直接用委托),當狀態改變時A去觸發這個事件。而B直接訂閱這個事件
 

這種設計有點問題
B由於要訂閱A的事件,所以B得完全引用A,其實有時候沒必要,因為我只關心A的狀態變化而已
狀態變更通知這種場景很多,有沒有更通用的方式呢?

 

解決思路

有個誰說的碰到問題加個中間層就解決了,如果解決不了就再加一層

A和B都引用ChangeToken,
B向ChangeToken註冊一個委托說:將來你有變化時回調我這個委托
當A的狀態變化時會調用ChangeToken的一個方法,這個方法內部就會去觸發執行B之前塞進去的委托
此時比如有組件C、D、E..都關心A的狀態變化,也可以引用ChangeToken,並向其註冊自己的委托
這樣ChangeToken可以作為一個通用組件,在很多需要更改通知是場景中使用,如:asp.net core的配置系統、終結點路由、 ....

 

實現

微軟定義了一個IChangeToken
HasChanged:表示當前這個ChangeToken是否變化過了
ActiveChangeCallbacks:當 A觸發ChangeToken發生變化時是否主動回調B塞進來的委托
RegisterChangeCallback(Action<object> callback, object state):提供一個方法,允許調用方塞入委托,B就是調用這個方法向ChangeToken塞入委托的。有一種情況是B希望在塞入委托的同時附帶一個狀態對象,將來委托被執行時這個狀態對象作為執行委托的參數

CancellationChangeToken是一個用的比較多的實現類,它包含一個CancellationToken屬性,這個屬性是通過構造函數來初始化的(CancellationTokenSource、CancellationToken自行查詢相關資料),
簡化的源碼如下:

 1 public class CancellationChangeToken : IChangeToken
 2 {
 3         public CancellationChangeToken(CancellationToken cancellationToken)
 4         {
 5             Token = cancellationToken;
 6         }
 7 
 8         public bool ActiveChangeCallbacks { get; private set; } = true;
 9 
10         public bool HasChanged => Token.IsCancellationRequested;
11 
12         private CancellationToken Token { get; }
13 
14         public IDisposable RegisterChangeCallback(Action<object> callback, object state)
15         {
16               return Token.Register(callback, state);           
17         }
18     }

因為CancellationToken天然的已經現了IChangeToken,因此CancellationChangeToken只是對CancellationToken的包裝。那為啥不直接讓CancellationToken實現IChangeToken呢?我感覺是設計意圖不同,CancellationToken設計時主要是考慮應用在取消非同步操作這件事上的,只是碰巧取消非同步操作與更改通知設計思路是相似的,所以才出現CancellationChangeToken

其它的實現類沒去研究過,但是只要你對這種設計思路理解了,碰到其它實現類應該看看就明白了

例子

下麵我們使用CancellationChangeToken來完成上面的A、B類,A類狀態變化時 通知到B類(其實就是執行B類忘ChangeToken中塞入的委托),完整源碼如下:

 1     class Program
 2     {
 3         static void Main(string[] args)
 4         {
 5             Console.OutputEncoding = Encoding.UTF8;
 6             CancellationTokenSource cts = new CancellationTokenSource();
 7             CancellationChangeToken cct = new CancellationChangeToken(cts.Token);
 8             var a = new A(cts);
 9             var b = new B(cct);
10             Console.ReadKey();
11         }
12     }
13 
14     public class A
15     {
16         CancellationTokenSource _cts;
17         public A(CancellationTokenSource cts)
18         {
19             this._cts = cts;
20             Task.Run(() =>
21             {
22                 Task.Delay(2000).Wait();
23                 Console.WriteLine("模擬觸發更改通知");
24                 _cts.Cancel();
25             });
26         }
27     }
28     public class B
29     {
30         public B(CancellationChangeToken cct) {
31             object testState = 1;
32             cct.RegisterChangeCallback(obj => {
33                 //將來cct檢測到變化時此委托會被執行
34                 //obj是註冊委托是傳遞進來的參數,就是這裡的testState
35                 Console.WriteLine($"狀態變化了,狀態值{obj}");
36             }, testState);
37         }
38     }

上面只是演示IChangeToken的思路,asp.net core中源碼的應用時通常是在A中提供一個返回IChangeToken的方法

無限監控與ChangeToken.OnChange

上面的方式只能變更通知一次,下麵可以永遠監控

 1     class Program
 2     {
 3         static void Main(string[] args)
 4         {
 5             Console.OutputEncoding = Encoding.UTF8;
 6             var a = new A();
 7             //實現無限監控狀態變化。OnChange有兩個委托類型的參數,我們分別稱為委托1和委托2
 8             ChangeToken.OnChange(() => a.CreateChangeToken(), () => Console.WriteLine("a狀態變化了"));
 9             Console.ReadKey();
10         }
11     }
12 
13     public class A
14     {
15         CancellationTokenSource _cts;
16         public A()
17         {
18             Task.Run(() =>
19             {
20                 while (true)
21                 {
22                     Task.Delay(2000).Wait();
23                     Console.WriteLine("模擬兩秒一次觸發一次狀態修改通知");
24                     this._cts.Cancel();
25                 }
26             });
27         }
28 
29         public IChangeToken CreateChangeToken() {
30             _cts = new CancellationTokenSource();
31             return new CancellationChangeToken(_cts.Token);
32         }
33     }

重點是這句:ChangeToken.OnChange(() => a.CreateChangeToken(), () => Console.WriteLine("a狀態變化了"));
OnChange有兩個委托類型的參數,我們分別稱為委托1和委托2,當a的狀態變化後會執行委托2,之後會執行委托1,當a狀態又變化時又會執行委托2,之後執行委托1,如此往複實現無限監控 

這是今天學習的內容,可能理解得不是很準確,僅供參考...


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

-Advertisement-
Play Games
更多相關文章
  • 第一種註冊方法 from flask import Flask app = Flask(__name__) @app.route("/hello") # 第一種註冊方法 def hello(): return "hello python!!!" if __name__ == "__main__": ...
  • 我們在進行APP開發的時候都會遇到一個文件:AndroidManifest.xml。從剛開始進行Android開發,到現在已經過去了幾個月,還是對這個文件一知半解,只知道它是配置用的。但是這文件里的東西具體有什麼用,該怎麼用一直都沒有理解。藉著做項目的機會,仔細研究一下這個文件。 研究Android ...
  • 這次我們可以看看併發中鎖的原理,大概會說到AQS,ReentrantLock,ReentrantReadWriteLock以及JDK8中新增的StampedLock,這些都是在java併發中很重要的東西,慢慢看吧! 一.LockSupport工具類 LockSupport工具類是jdk中rt.jar ...
  • 一、現象描述 筆者在用visual studio2010進行控制台程式進行程式編譯時候,經常會遇到代碼代碼沒有語法錯誤,但是編譯不通過的現象。系統報錯為 這個錯誤總是出現,特別是在每次新裝系統後,很是煩人。 二、出現原因 通過查閱相關資料可知: 出現該問題的原因通常是由於高版本與低版本之間的鏈接文件 ...
  • Redis詳解(一)——RDB 前言 由於 Redis 是一個記憶體資料庫,所謂記憶體資料庫,就是將資料庫中的內容保存在記憶體中,這與傳統的MySQL,Oracle等關係型資料庫直接將內容保存到硬碟中相比,記憶體資料庫的讀寫效率比傳統資料庫要快的多(記憶體的讀寫效率遠遠大於硬碟的讀寫效率)。但是保存在記憶體中也 ...
  • 項目簡介 項目來源於: "https://gitee.com/suimz_admin/BookShop" 一個基於JSP+Servlet+Jdbc的書店系統。涉及技術少,易於理解,適合JavaWeb初學者學習使用。 本人親測可正常啟動。 技術棧 前端技術 基礎:html+css+JavaScript ...
  • 轉發、重定向到其它業務方法 @org.springframework.stereotype.Controller @RequestMapping("/userController") public class UserController{ @RequestMapping("/handler1") ...
  • " 返回《C 併發編程》" "1. 調度到線程池" "2. 任務調度器" "2.1. Default 調度器" "2.2. 捕獲當前同步上下文 調度器" "2.3. ConcurrentExclusiveSchedulerPair 調度器" "3. 調度並行代碼" "4. 用調度器實現數據流的同步" ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...