EventBus In eShop -- 解析微軟微服務架構Demo(四)

来源:http://www.cnblogs.com/inday/archive/2017/06/29/eventbus-in-eshopcontainers.html
-Advertisement-
Play Games

引言大家好像對分析源碼厭倦了,說實在我也會厭倦,不過不看是無法分析其後面的東西,從易到難是一個必要的過程。今天說下EventBus,前幾天園裡的大神已經把其解刨,我今天就藉著大神的肩膀,分析下在eShop項目中EventBus的實現。最近發覺轉發文章不寫出處的,特此加上鏈接:http://inday... ...


引言

大家好像對分析源碼厭倦了,說實在我也會厭倦,不過不看是無法分析其後面的東西,從易到難是一個必要的過程。

今天說下EventBus,前幾天園裡的大神已經把其解刨,我今天就藉著大神的肩膀,分析下在eShop項目中EventBus的實現。

最近發覺轉發文章不寫出處的,特此加上鏈接:http://inday.cnblogs.com

解析源碼

我們知道使用EventBus是為瞭解除Publisher和Subscriber之間的依賴性,這樣我們的Publisher就不需要知道有多少Subscribers,只需要通過EventBus進行註冊管理就好了,在eShop項目中,有一個這樣的介面IEventBus(eShopOnContainers\src\BuildingBlocks\EventBus\EventBus\Abstractions)

public interface IEventBus
{
    void Subscribe<T, TH>(Func<TH> handler)
        where T : IntegrationEvent
        where TH : IIntegrationEventHandler<T>;
    void Unsubscribe<T, TH>()
        where TH : IIntegrationEventHandler<T>
        where T : IntegrationEvent;

    void Publish(IntegrationEvent @event);
}

我們可以看到這個介面定義了EventBus所需的一些操作, 對比大神的EventBus,相關功能都是一致的,我們看下它的實現類:EventBusRabbitMQ,從名字上可以看出,這是一個通過RabbitMQ來進行管理的EventBus,我們可以看到它使用了IEventBusSubscriptionsManager進行訂閱存儲,也就是大神文中的:

private readonly ConcurrentDictionary<Type, List<Type>> _eventAndHandlerMapping;

微軟在Demo中把其提取出了介面,把一些常用方法給提煉了出來,但是核心還是Dictionary<string, List<Delegate>>, 使用Dictionary進行Map映射。通過Subscribe和UnSubscribe進行訂閱和取消,使用Publish方法進行發佈操作。

public void Subscribe<T, TH>(Func<TH> handler)
    where T : IntegrationEvent
    where TH : IIntegrationEventHandler<T>
{
    var eventName = typeof(T).Name;
    var containsKey = _subsManager.HasSubscriptionsForEvent<T>();
    if (!containsKey)
    {
        if (!_persistentConnection.IsConnected)
        {
            _persistentConnection.TryConnect();
        }

        using (var channel = _persistentConnection.CreateModel())
        {
            channel.QueueBind(queue: _queueName,
                                exchange: BROKER_NAME,
                                routingKey: eventName);
        }
    }

    _subsManager.AddSubscription<T, TH>(handler);

}

我們看到在訂閱的時候,EventBus會檢查下在Map中是否有相應的註冊,如果沒有的話首先回去RabbitMQ中創建一個新的channel進行綁定,隨後在Map中進行註冊映射。

UnSubscribe則直接從Map中取消映射,通過OnEventRemoved事件判斷Map下此映射的subscriber是否為空,為空則從RabbitMQ中關閉channel。

在RabbitMQ的構造方法中,我們看到這樣一個創建:CreateConsumerChannel(),這裡創建了一個EventingBasicConsumer,當Queue中有新的消息時會通過ProcessEvent執行Map中註冊的handler(subscribers),看圖可能更清晰些:


Event In eShop

在ProcessEvent方法中,回去Map中找尋subscribers,然後通過動態反射進行執行:

private async Task ProcessEvent(string eventName, string message)
{

    if (_subsManager.HasSubscriptionsForEvent(eventName))
    { 
        var eventType = _subsManager.GetEventTypeByName(eventName);
        var integrationEvent = JsonConvert.DeserializeObject(message, eventType);
        var handlers = _subsManager.GetHandlersForEvent(eventName);

        foreach (var handlerfactory in handlers)
        {
            var handler = handlerfactory.DynamicInvoke();
            var concreteType = typeof(IIntegrationEventHandler<>).MakeGenericType(eventType);
            await (Task)concreteType.GetMethod("Handle").Invoke(handler, new object[] { integrationEvent });
        }
    }
}

微軟通過簡單的代碼解耦了Publisher和Subscribers之間的依賴關係,我們引用大神的總結:

image

應用

在catalog.api中,微軟出現了EventBus,我在上一篇中也提到了,這是我的一個疑惑,因為在catalog中並沒有訂閱操作,直接執行了Publish操作,原先以為是一個空操作,後來看了Basket.Api我才知道為何微軟要用RabbitMQ。

使用RabbitMQ,我們不僅是從類之間的解耦,更可以跨項目,跨語言,跨平臺的解耦,publisher僅僅需要把消息體(IntegrationEvent)傳送到RabbitMQ,Consumer從Queue中獲取消息體,然後推送到Subscribers執行相應的操作。我們看下Basket.Api.Startup.cs:

protected virtual void ConfigureEventBus(IApplicationBuilder app)
{
    var catalogPriceHandler = app.ApplicationServices
        .GetService<IIntegrationEventHandler<ProductPriceChangedIntegrationEvent>>();

    var orderStartedHandler = app.ApplicationServices
        .GetService<IIntegrationEventHandler<OrderStartedIntegrationEvent>>();

    var eventBus = app.ApplicationServices.GetRequiredService<IEventBus>();

    eventBus.Subscribe<ProductPriceChangedIntegrationEvent, ProductPriceChangedIntegrationEventHandler>
        (() => app.ApplicationServices.GetRequiredService<ProductPriceChangedIntegrationEventHandler>());

    eventBus.Subscribe<OrderStartedIntegrationEvent, OrderStartedIntegrationEventHandler>
        (() => app.ApplicationServices.GetRequiredService<OrderStartedIntegrationEventHandler>());
}

在這個方法里,我們看到了Subscribe操作,想想之前的提問有點搞笑,不過研究明白了也不錯,對吧!

總結

今天我們看了EventBus在Demo中的應用,總結一下。

1、EventBus可以很好的解耦訂閱者和發佈者之間的依賴

2、使用RabbitMQ能夠跨項目、跨平臺、跨語言的解耦訂閱者和發佈者

雖然在Demo中我們看到對訂閱者的管理是通過Dictionary記憶體的方式,所以我們的Subscribe僅僅只在Basket.Api中看到,但微軟是通過IEventBusSubscriptionsManager介面定義的,我們可以通過自己的需求來進行定製,可以做成分散式的,比如使用memcached。

寫在最後

每個月到下旬就會比較忙,所以文章發佈會比較慢,但我也會堅持學習完eShop的,為了學習,我建了個群,大家可以進來一起學習,有什麼建議和問題都可以進來哦。

eShop雖好,但不建議大家放到生產環境,畢竟是一個Demo,而且目前還是ALPHA版本,用來學習是一個很好的教材,這就是一個大雜燴,學習中你會學到很多新的東西,大家如果看好core的發展,可以一起研究下。

QQ群:376248054


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

-Advertisement-
Play Games
更多相關文章
  • Linux下的搜狗輸入法安裝: 1.搜狗官網下載Linux64bit版本文件,預設在/home/username/Downloads目錄下。 2.cd /home/username/Downloads/ & sudo dpkg -i sogoufilename.deb 3.sudo apt-get ...
  • 網路組team:是將多個網卡聚合在一起,從而實現容錯和提高吞吐量 1 創建網路組介面 nmcli connection add type team con-name TEAMname ifname INTname [config JSON] TEAMname 指定連接名,INname指定介面名, J ...
  • ip命令是Linux下較新的功能強大的網路配置工具。 1 功能 ip命令用來顯示或操縱Linux主機的路由、網路設備、策略路由和隧道。 2用法 Usage: ip [ OPTIONS ] OBJECT { COMMAND | help } ip [ -force ] -batch filename ...
  • 1.安裝此模塊需要先安裝sregex運行庫 apt-get update;apt-get install git make gcc -y #Centos改成yum git clone https://github.com/agentzh/sregex cd sregex make make inst ...
  • 之前在使用nginx和nginx-rtmp-module搭建流媒體伺服器的時候遇到一個很尷尬的問題,就是在把nginx-rtmp-module模塊添加到nginx中去的時候,我最開始採取的做法是先卸載原來的nginx,再下載nginx和nginx-rtmp-module的源碼重新編譯並安裝.重裝完之 ...
  • 在使用這種方法之前試過hdmi線連接,vga介面連接結果都不行。查找原因才知道,筆記本的hdmi和vga都只支持輸出,不支持輸入。 從msdn中的博客中得知該spacedesk(親測不支持xp,win7,8,10均可)http://spacedesk.ph/(官網)可以實現區域網內擴展屏幕。該軟體有 ...
  • 1.今天申請搭建了自己的雲虛擬主機 2.有挺多疑問:它只能使用ftp傳入主頁或者網站網頁內容嗎?沒有後臺代碼之類的怎麼實現程式的功能調用? ...
  • 最近在搞一個人臉識別的功能,使用了微軟的認知服務,一下講一個我遇到的小問題。 首先添加相關相應的NuGet:Microsoft.ProjectOxford.Face 然後構造FaceServiceClient並調用DetectAsync方法識別人臉信息。 構造FaceServiceClient需要一 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...