Entity Framework 6 Recipes 2nd Edition(9-7)譯->在WCF服務中序列化代理

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

9-7. 在WCF服務中序列化代理問題從一個查詢里返回一個動態代理對象,想要把它序列為一個POCO(Plain-Old CLR Objects)對象.實現基於POCO實體對象, 在運行時,EF會為每個實體自動生成一個派生類型,被稱為動態代理對象,代理對象會為POCO類重載很多虛擬屬性來註入執行操作的...


9-7. 在WCF服務中序列化代理

問題

從一個查詢里返回一個動態代理對象,想要把它序列為一個POCO(Plain-Old CLR Objects)對象.

實現基於POCO實體對象, 在運行時,EF會為每個實體自動生成一個派生類型,被稱為動態代理對象,

代理對象會為POCO類重載很多虛擬屬性來註入執行操作的掛鉤,像變更跟蹤,和延遲載入關聯的實體。

解決方案

假設我們有一個如Figure 9-7.所示的客戶模型

 

Figure 9-7. 客戶模型

我們將使用ProxyDataContractResolver類在服務端把一個代理對象反序列化為Client的POCO類

1. 創建Wcf服務應用程式.添加一個ADO.NET實體數據模型,並選擇”Client”表,創建好的模型,就如 Figure 9-7.所示.

2.打開Client的 POCO類, 為每個屬性添加virtual關鍵字,如Listing 9-33所示 . 這樣EF就可以創建動態代理類了。

============================================================================================

■■註意:如果你修改EDMX文件,EF會自動重新生成類,會重寫第2步里你對類的修改,你可以再次修改類或是修改T4模板來生成實體代碼。

=======================================================================

Listing 9-33. Our Client POCO Class and Our Object Vontext

    public partial class Client

    {

        public virtual int ClientId { get; set; }

        public virtual string Name { get; set; }

        public virtual string Email { get; set; }

}

3.我們需要為DataContractSerializer使用ProxyDataContractResolver類為WCF服務的客戶端把client 代理轉換為client實例.為此我們將創建一個操作行為特性 ,並且讓GetClient() 方法使用這個特性。新特性的代碼如 Listing 9-34 所示.註意:ProxyDataContractResolver 類屬於EF命名System.Data.Entity.Core.Objects

Listing 9-34. Our Custom Operation Behavior Attribute

namespace Recipe7

{

    public class ApplyProxyDataContractResolverAttribute :

        Attribute, IOperationBehavior

    {

        public void AddBindingParameters(OperationDescription description,

            BindingParameterCollection parameters)

        {

 

        }

 

        public void ApplyClientBehavior(OperationDescription description, ClientOperation proxy)

        {

            DataContractSerializerOperationBehavior dataContractSerializerOperationBehavior =

                description.Behaviors.Find<DataContractSerializerOperationBehavior>();

            dataContractSerializerOperationBehavior.DataContractResolver =

                new ProxyDataContractResolver();

        }

        public void Validate(OperationDescription description)

        {

 

        }

 

        public void ApplyDispatchBehavior(OperationDescription operationDescription, DispatchOperation dispatchOperation)

        {

            DataContractSerializerOperationBehavior

                dataContractSerializerOperationBehavior =

                operationDescription.Behaviors.Find<DataContractSerializerOperationBehavior>();

            dataContractSerializerOperationBehavior.DataContractResolver = new ProxyDataContractResolver();

        }

    }

}

4.用Listing 9-35里的代碼修改IService1.cs 介面

Listing 9-35. Our IService1 Interface Definition, Which Replaces the Code in IService1.cs

    [ServiceContract]

    public interface IService1

    {

 

        [OperationContract]

        void InsertTestRecord();

 

        [OperationContract]

        Client GetClient();

 

        [OperationContract]

        void Update(Client client);

    }

5. 用Listing 9-36里的代碼修改Service1.svc.cs 文件來實現服務介面。

Listing 9-36. The Implementation of the IService1 Interface, Which Replaces the Code in IService1.svc.cs

   public class Service1 : IService1

    {

 

        public void InsertTestRecord()

        {

            using (var context = new EFRecipesEntities())

            {

                //刪除之前的測試數據

                context.Database.ExecuteSqlCommand("delete from chapter9.client");

                //插入新的測試數據

                context.Database.ExecuteSqlCommand(@"insert into chapter9.client(name,email)values('Jerry Jones','[email protected]')");

            }

        }

 

        [ApplyProxyDataContractResolver]

        public Client GetClient()

        {

            using (var context = new EFRecipesEntities())

            {

                context.Configuration.LazyLoadingEnabled = false;

                return context.Clients.Single();

            }

        }

 

        public void Update(Client client)

        {

            using (var context = new EFRecipesEntities())

            {

                context.Entry(client).State = EntityState.Modified;

                context.SaveChanges();

            }

        }

}

6.在解決方案中添加一個新的Windows控制台應用程式,這是我們用來測試的客戶端,代碼如

Listing 9-37所示,添加WCF的引用。

Listing 9-37. Our Windows console application test client

    class Program

    {

        static void Main(string[] args)

        {

            using (var serviceClient=new ServiceReference1.Service1Client())

            {

                serviceClient.InsertTestRecord();

                var client = serviceClient.GetClient();

                Console.WriteLine("Client is :{0} at {1}",client.Name,client.Email);

                client.Name = "Alex Park";

                client.Email = "[email protected]";

                serviceClient.Update(client);

                client = serviceClient.GetClient();

                Console.WriteLine("Client changed to: {0} at {1}",client.Name, client.Email);

                Console.WriteLine("\npress any key to exit...");

                Console.ReadKey(true);

            }

        }

}

以下是控制台輸出結果:

===================================================================

Client is: Jerry Jones at [email protected]

Client changed to: Alex Park at [email protected]

=================================================================================

它是如何工作的?

微軟建議為WCF使用POCO對象,方便序列化實體對象。如果我們的應用程式使用POCO對象,並支持變更通知(把屬性設為virtual and導航對象集合類型為ICollection), EF會為從查詢返回的實體創建動態代理。這裡有兩個關於動態代理和WCF的問題,第一個問題是:必須序列化代理。 而DataContractSerializer 只能序列化和反序列化已知的類型,例如我們例子中的Client實體.然而 EF為Client 實體自動生成一個動態代理類,我們需要序列化這個代理類, 而不是 Client類,DataContractResolver就是解決這個問題的. 它能在序列化期間把一個類型映射到另一個類型. ProxyDataContractResolver 來源於DataContractResolver 並映射代理類型到 POCO類, 例如我們的 Client 實體. 為了使用ProxyDataContractResolver, 我們創建特性 (見 Listing 9-34) 來解決代理轉換成POCO類.我們在GetClient()方法上應用這個特性 (見Listing 9-36). 這樣Client實體的動態代理能正確的序列化,並被GetClient() 返回給WCF服務的調用者。第二個問題:必須處理延遲載入的問題.當  DataContractSerializer 序列化實體時, 它訪問實體的每個屬性,這會觸發延遲載入導航屬性. 這當然不是我們所希望的,所以我們要關閉延遲載入,如Listing 9-36所示。

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

 


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

-Advertisement-
Play Games
更多相關文章
  • 有時候,向服務端請求一個實體,我們希望返回如下的格式:links: [ href: http://localhost:8901/api/user/diaries/2013-08-17, rel: "self", method: "GET", isTemplated: false],currentDa...
  • 1. 用Response.Write方法代碼如下:Response.Write("");此方法缺陷就 是不能調用腳本文件中的自定義的函數,只能調用內部函數,具體調用自定義的函數只能在Response.Write寫上函數定義。比如:Response.Write("");2.用ClientScript類...
  • Console.WriteLine("start:出自http://www.cnblogs.com/ahjesus); List sourceArr = new List { 99, 51, 61, 41, 8, 73, -3, 225, 3, 10, 0 }; ...
  • 1. 用Response.Write方法 代碼如下: Response.Write(""); 此方法缺陷就是不能調用腳本文件中的自定義的函數,只能調用內部函數,具體調用自定義的函數只能在Response.Write寫上函數定 義,比如 Response.Write("");2.用Client...
  • 雖然我已經可以使用很多種編程語言進行工作,但我的工作常常會要求我快速掌握一門新的語言。我沒有選擇去閱讀幾百頁的程式手冊,而是快速瀏覽10到15頁的教程(可以在Google中搜索),並把程式語言的語法參考說明印在小卡片上(在google里搜索language to learn+reference ca...
  • C#導出PDF功能是開發中經常遇到的功能,我們採用第三方的組件,比如iTextSharp,aspose等,還能搜到一些開源的類庫,但是對於一些內容複雜樣式豐富的PDF,我們希望通過傳入一個URL直接生成一個PDF,並且不能與網頁原版差異太大,Pechkin倒是不錯,相對來說差異很小。在 Nuget ...
  • 存儲過程存儲過程一直存在於任何一種關係型資料庫中,如微軟的SQL Server.存儲過程是包含在資料庫中的一些代碼,通常為數據執行一些操作,它能為數據密集型計算提高性能,也能執行一些為業務邏輯. 當你使用數據的時候,有時你會通過存儲過程來獲取它們.在本章, 我們探討一些EF在使用存儲過程時,需要關註...
  • 通常情況下Server是無狀態的,在ASP.NET Web API中,我們可以讓服務端響應體中產生ETag屬性,起到緩存的作用。大致實現原理是:1、服務端的響應體中返回一個ETag屬性2、客戶端通過If-None-Match屬性把ETag的屬性值傳遞給服務端3、服務端返回304狀態碼響應體中返回ET...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...