使用SignalR和SQLTableDependency跟蹤資料庫中記錄的變動

来源:https://www.cnblogs.com/liujianshe1990-/archive/2019/02/20/10406252.html
-Advertisement-
Play Games

SqlTableDependency是一個組件用來接收資料庫的通知,包含在數據表上該記錄的值的Inserted、Deleted或者Update操作. 備註:原文提供示例代碼下載,但是js寫的有些問題(沒有更新數據),可以參考下文代碼修改一下,修改後的代碼可以接收插入和更新兩個,對刪除沒有處理。 介紹 ...


SqlTableDependency是一個組件用來接收資料庫的通知,包含在數據表上該記錄的值的Inserted、Deleted或者Update操作.

備註:原文提供示例代碼下載,但是js寫的有些問題(沒有更新數據),可以參考下文代碼修改一下,修改後的代碼可以接收插入和更新兩個,對刪除沒有處理。

介紹

SqlDependency 是用來接收數據表中指定的結果集在insert、update 或者delete 操作而發生變化的通知的一個類庫.不過,這個類不會回送記錄更改的值.所以,假如我想在web頁面中展示股票的值,收到每個通知後,我們都需要執行一個新的查詢來刷新緩存並刷新瀏覽器.如果我們股票值一發生變化瀏覽器就立馬顯示新的值,而不需要刷新瀏覽器,理想情況下我們想從web伺服器中接收通知,而不是從瀏覽器進行輪詢和從資料庫拉取數據.

解決方案是使用SignalR結合SqlTableDependency來處理; SqlTableDependency從資料庫獲取通知,接著使用SignalR給web頁面發送通知.

 

增強實現

TableDependency 是一個SQLDependency增強版的開源C#組件,當指定的表內容更改後用來發送事件。這個事件報告操作類型((INSERT/UPDATE/DELETE)以及變化刪除、插入或修改的值。組件的實現包含:

  • SqlTableDependency for SQL Server
  • OracleTableDependency for Oracle

TableDependency可以通過Nuget來進行載入。

如何工作

當實例化時,動態生成組件對象用來監視數據表的所有資料庫對象。在SqlTableDependency中,包含:

· Message Types

· Message Contract

· Queue

· Service Broker

· Table Trigger

· Stored Procedure

在應用程式突然退出情況下,用來清理創建的對象(也就是說,當應用程式終止時,沒有處理SqlTableDependency對象)

資料庫中生成的內容截圖:

clip_image002

所有這些對象會在SqlTableDependency 釋放的時候一次性釋放.

監視器

SqlTableDependency 內有一個watchDogTimeOut 對象,負責在程式突然斷開的時候移除對象。預設的超時時間是3分鐘,在發佈階段,這個時間還可以增加

通過上述的一系列對象,當表內容變化時,SqlTableDependency 獲取到通知並且發送包含記錄值的通知到C#事件.

代碼

假設有一個股票值的表,裡面的股票價格會頻繁變化:

CREATE TABLE [dbo].[Stocks](
 
[Code] [nvarchar](50) NULL,
 
[Name] [nvarchar](50) NULL,
 
[Price] [decimal](18, 0) NULL
 
) ON [PRIMARY]

我們把數據表的列映射到下麵的model:

public class Stock
{
    public decimal Price { get; set; }
    public string Symbol { get; set; }
    public string Name { get; set; }
}

接下來,需要使用Nuget安裝程式包:

PM> Install-Package SqlTableDependency

下一步,創建一個SignlaR的Hub類,繼承與SignalR的Hub類:

 

[HubName("stockTicker")]
public class StockTickerHub : Hub
{
    private readonly StockTicker _stockTicker;
 
    public StockTickerHub() : this(StockTicker.Instance)
    {
    }
 
    public StockTickerHub(StockTicker stockTicker)
    {
        _stockTicker = stockTicker;
    }
 
    public IEnumerable<Stock> GetAllStocks()
    {
        return _stockTicker.GetAllStocks();
    }
 
    public void alertAll()
    {
        Clients.All.testSignalR();
    }
}

我們將使用SignalR Hub API來處理伺服器端-客戶端的交互。StockTickerHub 類派生自SignalR的Hub類,用來處理客戶端的連接和方法調用。不能把這些方法放在Hub類裡面,因為Hub 的實例的生命周期為transient短暫的)。一個Hub 類會為每一個客戶端連接和方法調用創建實例。所以要保存股票數據,更新價格和廣播更新價格需要運行在一個獨立的類,這裡命名為StockTicker:

 

 
    public class StockTicker
    {
        // Singleton instance
        private readonly static Lazy<StockTicker> _instance = new Lazy<StockTicker>(
            () => new StockTicker(GlobalHost.ConnectionManager.GetHubContext<StockTickerHub>().Clients));
 
        private static SqlTableDependency<Stock> _tableDependency;
 
        private StockTicker(IHubConnectionContext<dynamic> clients)
        {
            Clients = clients;
 
            var mapper = new ModelToTableMapper<Stock>();
            mapper.AddMapping(s => s.Symbol, "Code");
 
            var connStr = ConfigurationManager.ConnectionStrings["connectionString"].ConnectionString;
 
            //此方法有11個重載,可以只指定連接字元串和表名
            _tableDependency = new SqlTableDependency<Stock>(connStr, "Stocks", mapper);
 
            _tableDependency.OnChanged += SqlTableDependency_Changed;
            _tableDependency.OnError += SqlTableDependency_OnError;
            _tableDependency.Start();
        }
 
        public static StockTicker Instance
        {
            get
            {
                return _instance.Value;
            }
        }
 
        private IHubConnectionContext<dynamic> Clients
        {
            get;
            set;
        }
 
        public IEnumerable<Stock> GetAllStocks()
        {
            var stockModel = new List<Stock>();
 
            var connectionString = ConfigurationManager.ConnectionStrings["connectionString"].ConnectionString;
            using (var sqlConnection = new SqlConnection(connectionString))
            {
                sqlConnection.Open();
                using (var sqlCommand = sqlConnection.CreateCommand())
                {
                    sqlCommand.CommandText = "SELECT * FROM [Stocks]";
 
                    using (var sqlDataReader = sqlCommand.ExecuteReader())
                    {
                        while (sqlDataReader.Read())
                        {
                            var code = sqlDataReader.GetString(sqlDataReader.GetOrdinal("Code"));
                            var name = sqlDataReader.GetString(sqlDataReader.GetOrdinal("Name"));
                            var price = sqlDataReader.GetDecimal(sqlDataReader.GetOrdinal("Price"));
 
                            stockModel.Add(new Stock { Symbol = code, Name = name, Price = price });
                        }
                    }
                }
            }
 
            return stockModel;
        }
 
        private void SqlTableDependency_OnError(object sender, ErrorEventArgs e)
        {
            throw e.Error;
        }
 
        /// <summary>
        /// Broadcast New Stock Price
        /// </summary>
        private void SqlTableDependency_Changed(object sender, RecordChangedEventArgs<Stock> e)
        {
            if (e.ChangeType != ChangeType.None)
            {
                BroadcastStockPrice(e.Entity);
            }
        }
 
        private void BroadcastStockPrice(Stock stock)
        {
            Clients.All.updateStockPrice(stock);
        }
 
        #region IDisposable Support
 
        private bool disposedValue = false; // To detect redundant calls
 
        protected virtual void Dispose(bool disposing)
        {
            if (!disposedValue)
            {
                if (disposing)
                {
                    _tableDependency.Stop();
                }
 
                disposedValue = true;
            }
        }
 
        ~StockTicker()
        {
            Dispose(false);
        }
 
        // This code added to correctly implement the disposable pattern.
        public void Dispose()
        {
            Dispose(true);
            GC.SuppressFinalize(this);
        }
 
        #endregion IDisposable Support
    }

 

現在來看一下HTML頁面:

<!DOCTYPE html>
<html>
<head>
    <title>SqlTableDependencly with SignalR</title>
    <link href="StockTicker.css" rel="stylesheet" />
</head>
<body>
    <h1>SqlTableDependencly with SignalR</h1>
 
    <input type="button" value="測試SignalR" id="btnTest" />
    <div id="stockTable">
        <table border="1">
            <thead style="
                <tr><th>Code</th><th>Name</th><th>Price</th></tr>
            </thead>
            <tbody>
                <tr class="loading"><td colspan="3">loading...</td></tr>
            </tbody>
        </table>
    </div>
 
    <script src="jquery-1.10.2.min.js"></script>
    <script src="jquery.color-2.1.2.min.js"></script>
    <script src="../Scripts/jquery.signalR-2.2.0.js"></script>
    <script src="../signalr/hubs"></script>
    <script src="SignalR.StockTicker.js"></script>
</body>
</html>

下麵是js處理SignalR中方法調用和返回的數據的代碼:

/// <reference path="../Scripts/jquery-1.10.2.js" />
/// <reference path="../Scripts/jquery.signalR-2.1.1.js" />
 
// Crockford's supplant method (poor man's templating)
if (!String.prototype.supplant) {
    String.prototype.supplant = function (o) {
        return this.replace(/{([^{}]*)}/g,
            function (a, b) {
                var r = o[b];
                return typeof r === 'string' || typeof r === 'number' ? r : a;
            }
        );
    };
}
 
$(function () {
    var ticker = $.connection.stockTicker; // the generated client-side hub proxy
    var $stockTable = $('#stockTable');
    var $stockTableBody = $stockTable.find('tbody');
    var rowTemplate = '<tr data-symbol="{Symbol}"><td>{Symbol}</td><td>{Name}</td><td>{Price}</td></tr>';
 
    $("#btnTest").click(function () {
        ticker.server.alertAll();
    });
 
    function formatStock(stock) {
        return $.extend(stock, {
            Price: stock.Price.toFixed(2)
        });
    }
 
    function init() {
        return ticker.server.getAllStocks().done(function (stocks) {
            $stockTableBody.empty();
 
            $.each(stocks, function () {
                var stock = formatStock(this);
                $stockTableBody.append(rowTemplate.supplant(stock));
            });
        });
    }
 
    // Add client-side hub methods that the server will call
    $.extend(ticker.client, {
        updateStockPrice: function (stock) {
            var displayStock = formatStock(stock);
            $row = $(rowTemplate.supplant(displayStock));
            var $oldRow = $stockTableBody.find('tr[data-symbol=' + stock.Symbol + ']');
            if ($oldRow.length) {
                $oldRow.replaceWith($row);
            } else {
                $stockTableBody.append($row);
            }
        }
    });
 
    $.extend(ticker.client, {
        testSignalR: function () {
            alert("伺服器發通知了");
        }
    });
 
    // Start the connection
    $.connection.hub.start().then(init);
});

最後,不要忘記在StartUp中註冊SignalR的路由:

[assembly: OwinStartup(typeof(Stocks.Startup))]
 
namespace Stocks
{
    public static class Startup
    {
        public static void Configuration(IAppBuilder app)
        {
            app.MapSignalR();
        }
    }
}

如何測試

在附件中,包含一個簡單的web應用,包含一個頁面,用來展示股票價格變動的表格.

如果進行測試,按如下步驟操作:

· 首先為目標資料庫執行語句:

ALTER DATABASE MyDatabase SET ENABLE_BROKER

· 創建數據表: 

CREATE TABLE [dbo].[Stocks](
[Code] [nvarchar](50) NOT NULL, 
[Name] [nvarchar](50) NOT NULL, 
[Price] [decimal](18, 0) NOT NULL
)

· 生成幾條數據到資料庫表.

· 運行web應用,打開 /SignalR.Sample/StockTicker.html 頁面.

· 修改數據表中的數據,可以看到表格中的數據會隨之更新.

補充

關於SqlTableDependency

微軟本身提供了一個數據變動通知的實現:SqlDependency,但如作者所說,只能得到變動通知,並不知道發生了什麼變化.SqlDependency支持的 SELECT 語句如下,詳細介紹:查看

滿足下列要求的 SELECT 語句支持查詢通知:

· 必須顯式說明 SELECT 語句中提取的列,並且表名必須限定為兩部分組成的名稱。註意,這意味著語句中引用的所有表都必須處於同一資料庫中。

· 語句不能使用星號 (*) 或 table_name.* 語法指定列。

· 語句不能使用未命名列或重覆的列名。

· 語句必須引用基表。

· 語句不能引用具有計算列的表。

· 在 SELECT 語句中提取的列不能包含聚合表達式,除非語句使用 GROUP BY 表達式。提供 GROUP BY 表達式時,選擇列表便可以包含聚合函數 COUNT_BIG() 或 SUM()。但是,不能為可為空的列指定 SUM()。語句不能指定 HAVING、CUBE 或 ROLLUP。

· 在用作簡單表達式的 SELECT 語句中提取的列不能多次顯示。

· 語句不能包含 PIVOT 或 UNPIVOT 運算符。

· 語句不能包含 UNION、INTERSECT 或 EXCEPT 運算符。

· 語句不能引用視圖。

· 語句不能包含下列任意一個:DISTINCT、COMPUTE、COMPUTE BY 或 INTO。

· 語句不能引用伺服器全局變數 (@@variable_name)。

· 語句不能引用派生表、臨時表或表變數。

· 語句不能從其他資料庫或伺服器中引用表或視圖。

· 語句不能包含子查詢、外部聯接或自聯接。

· 語句不能引用下列大型對象類型:textntext 和 image

· 語句不能使用 CONTAINS 或 FREETEXT 全文謂詞。

· 語句不能使用行集函數,包括 OPENROWSET 和 OPENQUERY。

· 語句不能使用下列任何一個聚合函數:AVG、COUNT(*)、MAX、MIN、STDEV、STDEVP、VAR 或 VARP。

· 語句不能使用任何具有不確定性的函數,包括排名函數和開窗函數。

· 語句不能包含用戶定義聚合。

· 語句不能引用系統表或視圖,包括目錄視圖和動態管理視圖。

· 語句不能包含 FOR BROWSE 信息。

· 語句不能引用隊列。

· 語句不能包含無法更改和無法返回結果的條件語句(如 WHERE 1=0)。

· 語句不能指定 READPAST 鎖提示。

· 語句不能引用任何 Service Broker QUEUE。

· 語句不能引用同義詞。

· 語句不能具有基於 double/real 數據類型的比較或表達式。

· 語句不得使用 TOP 表達式。

SqlTableDependency 是一個增強的.NET SqlDepenency其優勢在於包含了Insert、Update以及Delete的記錄的值,以及在表上執行的DML操作(Insert/Delete/Update)。這是與.NET SqlDepenency最大的差異,.NET SqlDepenency沒有告訴你哪些數據在資料庫上發生了更改。

PS: 如果想要使用資料庫通知,必須在資料庫中啟用Server Broker服務,可以執行如下語句: ALTER DATABASE MyDatabase SET ENABLE_BROKER
 
使用SqlTableDependency 的步驟:
1.   創建一個SqlTableDependency 實例,並傳入連接字元串、表名等參數
2.   訂閱SqlTableDependency 的OnChange事件
3.   調用Start()方法開始接收通知
4.   調用Stop()方法停止接收通知

引用

· SignalR: http://www.asp.net/signalr/overview/getting-started/tutorial-server-broadcast-with-signalr

· SqlTableDependency: https://tabledependency.codeplex.com/

· MSDN 主題:


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

-Advertisement-
Play Games
更多相關文章
  • 成功的故事固然蕩氣迴腸,失敗的故事也能千迴百轉,但是如果你連第一步都不願邁出,不去追求與償試,那麼你永完不會知道成功或者失敗的原因是什麼,人生一輩子也就三萬多天,如果每天糊裡糊塗的過去了,若幹年後可能連故事都沒得講。 年前的時候廣州.net俱樂部主席葉老師就曾微信找我私聊,說你可以在合肥組織一個合肥 ...
  • ASP.net core 使用UEditor.Core 實現 ueditor 上傳功能 首先通過nuget 引用UEditor.Core,作者github:https://github.com/baiyunchen/UEditor.Core/ 在Startup.cs中添加 引入ueditor並配置 ...
  • 一. 支持欄位 EF允許讀取或寫入欄位而不是一個屬性。在使用實體類時,用面向對象的封裝來限制或增強應用程式代碼對數據訪問的語義時,這可能很有用。無法使用數據註釋配置。除了約定,還可以使用Fluent API為屬性配置支持欄位。 1.1 約定 1.2 Fluent API 二. 構造函數 從開始 EF ...
  • 1.對微服務的理解 之前看到在群里的朋友門都在討論微服務,看到他們的討論,我也有了一些自己的理解,所謂微服務就是系統里的每個服務都 可以自由組合。自由組合這個就很厲害了,這樣一來,每個服務與服務之間基本的物理 耦合為0,橫向擴展整個系統就會非常非常靈活。 surging的厲害之處也恰恰是可以做到這些 ...
  • 鏈接:https://pan.baidu.com/s/1SJHKTvQ4av_P9kLrtyYZ6w 提取碼:toya ...
  • 右鍵“工具箱”選擇“選擇項” 彈出對話框 選擇“瀏覽” 選中下載還的dll庫文件。完成後工具箱中會有 treeGridView控制項,如下圖 拖動控制項到windform 中 添加代碼 參考: https://www.cnblogs.com/mrtiny/p/5174095.html ...
  • Openfire簡介    Openfire 是開源的、基於可拓展通訊和表示協議(XMPP)、採用Java編程語言開發的實時協作伺服器。Openfire的效率很高,單台伺服器可支持上萬併發用戶。   Server和Client端的通信都用xml文檔的形式進行通 ...
  • 上一篇文章講了 k8s使用helm打包chart並上傳到騰訊雲TencentHub,今天就講一下使用Helm部署應用並使用configMap代替asp.net core 中的appsettings.json文件。 把Chart上傳到TencentHub之後,我們就可以通過騰訊雲的容器服務,直接部署H ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...