Entity Framework 6 Recipes 2nd Edition(9-6)譯->管理斷開時的併發

来源:http://www.cnblogs.com/kid1412/archive/2016/01/19/5142428.html
-Advertisement-
Play Games

9-6. 管理斷開時的併發問題想要確保只接受在WCF客戶端併發令牌未被修改的實體。解決方案我們有一個如Figure 9-6所示的模型.Figure 9-6訂單實體模型我們想通過WCF服務來更新一個order ,而要確保這個order 在我們上次取回後沒有發生過其它修改。稍後我們將演示兩個不同的方式....


9-6. 管理斷開時的併發

問題

想要確保只接受在WCF客戶端併發令牌未被修改的實體。

解決方案

我們有一個如Figure 9-6所示的模型.

 

Figure 9-6訂單實體模型

我們想通過WCF服務來更新一個order ,而要確保這個order 在我們上次取回後沒有發生過其它修改。

稍後我們將演示兩個不同的方式.兩種都用一個TimeStamp列作併發控制。

1. 新建一個WCF服務庫,右擊解決方案,選擇“新建項目”,選擇WCF ➤ WCF 服務庫,並命名為Recipe6.

2. 右擊這個項目,選擇“添加” ➤“新建項” ,選擇 數據➤ ADO.NET實體數據模型.在嚮導中選擇Order表. 在EF模型設計圖中,右擊TimeStamp 屬性, select “屬性“, 把“併發模式”設為Fixed.

3. 用Listing 9-30里的代碼定義IService1.cs 文件里的服務

Listing 9-30. Our WCF Service Contract

    [ServiceContract]

    public interface IService1

    {

        [OperationContract]

        Order InsertOrder();

        [OperationContract]

        void UpdateOrderWithoutRetrieving(Order order);

        [OperationContract]

        void UpdateOrderByRetrieving(Order order);

}

4. 用Listing 9-31的代碼實現Service1.cs 文件里的服務。

Listing 9-31. The Implementation of Our Service Contract

public class Service1 : IService1

    {

        public Order InsertOrder()

        {

            using (var context=new EFRecipesEntities())

            {

                //刪除之前的測試數據

                context.Database.ExecuteSqlCommand("delete from chapter9.[order]");

                var order = new Order { Product = "Camping Tent", Quantity = 3, Status = "Received" };

                context.Orders.Add(order);

                context.SaveChanges();

                return order;

            }

        }

 

        public void UpdateOrderWithoutRetrieving(Order order)

        {

            using (var context=new EFRecipesEntities())

            {

                try

                {

                    context.Orders.Attach(order);

                    if (order.Status=="Received")

                    {

                        context.Entry(order).Property(x => x.Quantity).IsModified = true;

                        context.SaveChanges();

                    }

                }

                catch (OptimisticConcurrencyException ex)

                {

                    //處理併發異常

                }

            }

        }

 

        public void UpdateOrderByRetrieving(Order order)

        {

            using (var context=new EFRecipesEntities())

            {

                //從資料庫中獲取當前實體

                var dbOrder = context.Orders.Single(o => o.OrderId == order.OrderId);

               //執行併發檢查

                if (dbOrder!=null&&StructuralComparisons.StructuralEqualityComparer.Equals(order.TimeStamp,dbOrder.TimeStamp))

                {

                    dbOrder.Quantity = order.Quantity;

                    context.SaveChanges();

                }

                else

                {

                    //處理併發問題

                }

            }

        }

}

5. 為測試服務,我們在解決方案里創建一個控制台應用程式.在服務項目上右擊,選擇“調試” ➤啟動新實例,然後在控制項目里添加對這個服務的引用。控制台項目代碼如Listing 9-32 所示.

Listing 9-32. The Client We Use to Test Our WCF Service

    class Program

    {

        static void Main(string[] args)

        {

            var service = new ServiceReference1.Service1Client();

            var order = service.InsertOrder();

            order.Quantity = 5;

            service.UpdateOrderWithoutRetrieving(order);

            order = service.InsertOrder();

            order.Quantity = 3;

            service.UpdateOrderByRetrieving(order);

        }

}

如果在Main() 方法前設置斷點,逐語句執行,將會看到代碼運行入服務端。

它是如何工作的?

InsertOrder() 方法(見Listing 9-31) 刪除之前的測試數據,插入一個新的order, 並且返回包含正確ID值和TimeStamp的order.在我們客戶端,我們用兩個略微不同的方式來更新這個order.

第一種用UpdateOrderWithoutRetrieving()方式:先把從客戶端傳過來的order 附加Attach()到DbContext,然後判斷order狀態是否為“Received”,如果是我們把實體 Quantity 屬性狀態設為 modified然後調用

SaveChanges(). EF會生成一個更新SqL語句,它的Where子句里包含OrderId和TimeStamp.如果SaveChanges()前資料庫中該條數據被修改過,則會產生異常(存儲區更新、插入或刪除語句影響到了意外的行數(0)。實體在載入後可能被修改或刪除。刷新 ObjectStateManager 項。)。我們用try,catch來捕獲異常. 這樣就可以保證客戶端提交過來的Order在運行InsertOrder() 之後,資料庫里對應記錄的數據未被修改過. 其實我們可以在保存前先進行併發檢查,從客戶端回傳的order的TimeStamp屬性值與資料庫的值比較。這就是UpdateOrderByRetrieving() 方法里採用的方式。

附:創建示例用到的資料庫的腳本文件


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

-Advertisement-
Play Games
更多相關文章
  • 存儲過程存儲過程一直存在於任何一種關係型資料庫中,如微軟的SQL Server.存儲過程是包含在資料庫中的一些代碼,通常為數據執行一些操作,它能為數據密集型計算提高性能,也能執行一些為業務邏輯. 當你使用數據的時候,有時你會通過存儲過程來獲取它們.在本章, 我們探討一些EF在使用存儲過程時,需要關註...
  • 通常情況下Server是無狀態的,在ASP.NET Web API中,我們可以讓服務端響應體中產生ETag屬性,起到緩存的作用。大致實現原理是:1、服務端的響應體中返回一個ETag屬性2、客戶端通過If-None-Match屬性把ETag的屬性值傳遞給服務端3、服務端返回304狀態碼響應體中返回ET...
  • 9-7. 在WCF服務中序列化代理問題從一個查詢里返回一個動態代理對象,想要把它序列為一個POCO(Plain-Old CLR Objects)對象.實現基於POCO實體對象, 在運行時,EF會為每個實體自動生成一個派生類型,被稱為動態代理對象,代理對象會為POCO類重載很多虛擬屬性來註入執行操作的...
  • 關於Ajax說法錯誤的是( )。(選擇一項)MVC是一種流行的軟體設計模式,它把系統分為三個模塊。三個模塊為( )。在ASP.NET中,關於WebService的說法正確的是( ).NET中Object類是所有類的基類,用戶自定義的類部預設地繼承這個類。以下選項不屬於Object類的方法的是( )。...
  • 訂單揀貨完成後,需要把訂單裝箱打包,並列印客戶地址信息。訂單打包的操作流程先是掃描訂單號,然後掃描商品條碼。 1.訂單打包 列印包裝箱面單 2.訂單發貨 訂單打包完成後就等待發貨,快遞公司來拉貨的時候,進行訂單發貨交接,訂單發貨界面如下
  • 本篇不涉及具體代碼,而是把實際開發UWP APP的過程中,遇到的不同設備,不同解析度顯示效果差異的問題進行討論。希望能夠拋磚引玉,和各位擦出一些火花。 蜀黍我目前是在做一套牛逼的UWP APP啦,目標是能跑在各種尺寸不同,解析度不同的PC,Phone和Tablet上。無論是從代碼的複雜度還是實...
  • 今天給大家分享的內容基於前幾天收到的一條留言信息,留言內容是這樣的:看了這位網友的留言相信有不少剛接觸開發的童鞋們也會有同樣的困惑,除了用新建類作為橋梁之外還有什麼好的辦法?很多童鞋肯定會想到DataSet,不錯,DataSet完全可以實現各種數據集的傳遞,而且還很方便,但是一般我們更推薦使用Lis...
  • TopShelf簡介 個人理解:開源、跨平臺的服務框架。提供一種方式以控制台編寫windows服務,與windows服務相比,目前只發現便於調試。 官網網站:http://docs.topshelf-project.com/en/latest/index.htmlTopShelf基本用法...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...