在B2B的项目中,一般用户注册后,就有一个属于自己的店铺,此时,我们就要插入两张表, User和Shop表。

当然,要么插入成功,要么全失败。

第一步: 首先看一下项目的结构图:

第二步: 准备工作,我们新建Commerce数据库,用EF去映射,然后新建ServiceWCF类库,具体步骤就省略,

这一块不懂可以留言。

第三步:新建一个Model类库。建立两个实体类Shop和User,当然自定义类型在WCF中传输,

必须在类上加上【DataContract】,属性上加【DataMember】。

Shop.cs

 1 namespace Model
 2 {
 3     [DataContract]
 4     public class Shop
 5     {
 6         [DataMember]
 7         public int ShopID { get; set; }
 8
 9         [DataMember]
10         public int UserID { get; set; }
11
12         [DataMember]
13         public string ShopName { get; set; }
14
15         [DataMember]
16         public string ShopUrl { get; set; }
17
18     }
19 }

User.cs

 1 namespace Model  2 {  3     [DataContract]  4     public class User  5     {  6         [DataMember]  7         public int UserID { get; set; }  8   9         [DataMember] 10         public string UserName { get; set; } 11  12         [DataMember] 13         public string Password { get; set; } 14     } 15 }

第四步:然后在ServiceWCF类库中新建两个文件Seller.cs 和 ISeller.cs.

ISeller.cs:

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Runtime.Serialization;
 5 using System.ServiceModel;
 6 using System.Text;
 7
 8 namespace ServiceWCF
 9 {
10     [ServiceContract]
11     public interface ISeller
12     {
13         [OperationContract(Name = "AddUser")]
14         bool Add(Model.User user, out int userID);
15
16         [OperationContract(Name = "AddShop")]
17         bool Add(Model.Shop shop, out int shopID);
18
19         [OperationContract]
20         bool Add(Model.User user, Model.Shop shop);
21     }
22 }

Seller.cs

 1 namespace ServiceWCF  2 {  3     public class Seller : ISeller  4     {  5         ///<summary>  6 /// User的插入操作  7 ///</summary>  8 ///<param name="user"></param>  9 ///<param name="userID"></param> 10 ///<returns></returns> 11         public bool Add(Model.User user, out int userID) 12         { 13             using (CommerceEntities db = new CommerceEntities()) 14             { 15                 try 16                 { 17                     User userModel = new User() 18                     { 19                         UserName = user.UserName, 20                         Passwrod = user.Password 21                     }; 22  23                     db.User.AddObject(userModel); 24  25                     db.SaveChanges(); 26  27                     userID = userModel.UserID; 28  29                     return true; 30                 } 31                 catch (Exception) 32                 { 33                     userID = 0; 34                     throw; 35                 } 36             } 37         } 38  39         ///<summary> 40 /// Shop的插入操作 41 ///</summary> 42 ///<param name="shop"></param> 43 ///<param name="shopID"></param> 44 ///<returns></returns> 45         public bool Add(Model.Shop shop, out int shopID) 46         { 47             using (CommerceEntities db = new CommerceEntities()) 48             { 49                 try 50                 { 51  52                     Shop shopModel = new Shop() 53                           { 54                               ShopName = shop.ShopName, 55                               ShopUrl = shop.ShopUrl, 56                               UserID = shop.UserID 57                           }; 58  59                     db.Shop.AddObject(shopModel); 60  61                     db.SaveChanges(); 62  63                     shopID = shopModel.ShopID; 64  65                     return true; 66                 } 67                 catch (Exception) 68                 { 69                     shopID = 0; 70                     throw; 71                 } 72             } 73         } 74  75 ///<summary> 76 /// User,Shop的插入的操作 77 ///</summary> 78 ///<param name="user"></param> 79 ///<param name="shop"></param> 80 ///<returns></returns>
81         [OperationBehavior(TransactionScopeRequired = true, TransactionAutoComplete = true)] 82         public bool Add(Model.User user, Model.Shop shop) 83         { 84             int shopID; 85             int UserID; 86  87             //注意,这个方法操作了两个数据库实例,为AddUser和AddShop。所以晋升为分布式事务 88             if (Add(user, out  UserID)) 89             { 90                 shop.UserID = UserID; 91  92                 return Add(shop, out shopID); 93             } 94  95             return false; 96         } 97     } 98 }

TransactionScopeRequired: 告诉ServiceHost自托管服务,进入我的方法,必须给我加上事务。

TransactionAutoComplete:   方法执行中,如果没有抛出异常,则自动提交。

第五步: 新建Host来承载了,配置AppConfig,这些细节就不再说了。

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Text;
 5
 6 namespace ServiceHost
 7 {
 8     class Program
 9     {
10         static void Main(string[] args)
11         {
12             System.ServiceModel.ServiceHost host = new System.ServiceModel.ServiceHost(typeof(ServiceWCF.Seller));
13
14             host.Open();
15
16             Console.WriteLine("WCF 服务已经开启!");
17
18             Console.Read();
19         }
20     }
21 }
 1 <?xml version="1.0" encoding="utf-8"?>  2 <configuration>  3   <system.web>  4     <compilation debug="true" />  5   </system.web>  6   <!-- 部署服务库项目时,必须将配置文件的内容添加到   7   主机的 app.config 文件中。System.Configuration 不支持库的配置文件。-->  8   <system.serviceModel>  9     <services> 10       <service name="ServiceWCF.Seller"> 11         <endpoint address="" binding="wsHttpBinding" contract="ServiceWCF.ISeller"> 12           <identity> 13             <dns value="localhost" /> 14           </identity> 15         </endpoint> 16         <endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" /> 17         <host> 18           <baseAddresses> 19             <add baseAddress="http://localhost:8732/Seller/" /> 20           </baseAddresses> 21         </host> 22       </service> 23     </services> 24     <behaviors> 25       <serviceBehaviors> 26         <behavior> 27           <!-- 为避免泄漏元数据信息, 28           请在部署前将以下值设置为 false 并删除上面的元数据终结点  --> 29           <serviceMetadata httpGetEnabled="True" /> 30           <!-- 要接收故障异常详细信息以进行调试, 31           请将以下值设置为 true。在部署前设置为 false  32             以避免泄漏异常信息--> 33           <serviceDebug includeExceptionDetailInFaults="False" /> 34         </behavior> 35       </serviceBehaviors> 36     </behaviors> 37   </system.serviceModel> 38   <connectionStrings> 39     <add name="CommerceEntities" connectionString="metadata=res://*/Commerce.csdl|res://*/Commerce.ssdl|res://*/Commerce.msl;provider=System.Data.SqlClient;provider connection string=&quot;Data Source=VONXCEVF0IT7JDJ;Initial Catalog=Commerce;Integrated Security=True;MultipleActiveResultSets=True&quot;" providerName="System.Data.EntityClient" /> 40   </connectionStrings> 41 </configuration>

第六步:开启WCF服务,新建ServiceClient类库,然后用信道生成实例。

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Text;
 5 using System.ServiceModel;
 6 using ServiceWCF;
 7
 8 namespace ServiceClient
 9 {
10     class Program
11     {
12         static void Main(string[] args)
13         {
14             var user = new Model.User()
15             {
16                 UserName = "huangxincheng520",
17                 Password = "i can fly"
18             };
19
20             var shop = new Model.Shop()
21             {
22                 ShopName = "shopex",
23                 ShopUrl = "http://www.shopex.cn"
24             };
25
26             var factory = new ChannelFactory<ISeller>(new WSHttpBinding(),
                                  new EndpointAddress("http://localhost:8732/Seller/"));
27
28             var client = factory.CreateChannel();
29
30             if (client.Add(user, shop))
31                 Console.WriteLine("huangxincheng520, 恭喜你,数据插入成功。");
32             else
33                 Console.WriteLine("huangxincheng520, 呜呜,数据插入失败。");
34
35             Console.Read();
36         }
37     }
38 }

最后就是测试了:

首先:走正常流程。client.Add方法调用服务器端,运行效果如图所示:

是的,数据已经正常插入成功,对Client端而言,这个操作是透明的。

然后:  我们在Seller类中的Add方法中故意加入异常。看效果咋样。

 1   ///<summary>  2 /// User,Shop的插入的操作  3 ///</summary>  4 ///<param name="user"></param>  5 ///<param name="shop"></param>  6 ///<returns></returns>  7         [OperationBehavior(TransactionScopeRequired = true, TransactionAutoComplete = true)]  8         public bool Add(Model.User user, Model.Shop shop)  9         { 10             int shopID; 11             int UserID; 12              13             if (Add(user, out  UserID)) 14             { 15                 //注意注意,我在Adduser成功的情况下,抛出异常,看是否回滚。 16                 throw new Exception(); 17  18                 shop.UserID = UserID; 19  20                 return Add(shop, out shopID); 21             } 22  23             return false; 24         }

截图如下:

哈哈,抛出异常了,我的Exception起到效果了,再来看一下数据库。大家都知道会发生什么了,对的,异常不再产生数据了,

还是先前产生了那条数据,说明起到效果了。

WCF事务应用[转]的更多相关文章

  1. WCF入门(十)---WCF事务

    事务处理在WCF(Windows Communication Foundation)是一套遵循一些性质,统称为ACID的操作.这里,如果一个操作出现故障,整个系统就会自动失败.如网上订单生成,就可能使 ...

  2. wcf事务(随记)

    ----------------------------------------------------wcf事务:1.ACID:原子性.一致性.隔离性.持久性:2.事务:添加命名空间(using S ...

  3. WCF分布式开发步步为赢(12):WCF事务机制(Transaction)和分布式事务编程

    今天我们继续学习WCF分布式开发步步为赢系列的12节:WCF事务机制(Transaction)和分布式事务编程.众所周知,应用系统开发过程中,事务是一个重要的概念.它是保证数据与服务可靠性的重要机制. ...

  4. wcf事务

    wcf服务 using System; using System.Collections.Generic; using System.Linq; using System.Runtime.Serial ...

  5. WCF学习笔记之事务编程

    WCF学习笔记之事务编程 一:WCF事务设置 事务提供一种机制将一个活动涉及的所有操作纳入到一个不可分割的执行单元: WCF通过System.ServiceModel.TransactionFlowA ...

  6. WCF基础之事务

    说到事务,我最先想到的是“回滚”. 百科:事务是恢复和并发控制的基本单位.事务应该具有4个属性:原子性.一致性.隔离性.持久性.这四个属性通常称为ACID特性.好了,具体的就不多复制了. 我小试了一下 ...

  7. 跟我一起学WCF(13)——WCF系列总结

    引言 WCF是微软为了实现SOA的框架,它是对微乳之前多种分布式技术的继承和扩展,这些技术包括Enterprise Service..NET Remoting.XML Web Service.MSMQ ...

  8. 跟我一起学WCF(10)——WCF中事务处理

    一.引言 好久没更新,总感觉自己欠了什么一样的,所以今天迫不及待地来更新了,因为后面还有好几个系列准备些,还有很多东西需要学习总结的.今天就来介绍下WCF对事务的支持. 二.WCF事务详解 2.1 事 ...

  9. C#综合揭秘——细说事务

    引言 其实事务在数据层.服务层.业务逻辑层多处地方都会使用到,在本篇文章将会为大家一一细说. 其中前面四节是事务的基础,后面的三节是事务的重点,对事务有基础的朋友可以跳过前面四节. 文章有错漏的地方欢 ...

随机推荐

  1. 整理几篇比较好的AndroidUI动画开发文章

    第一篇,写的比较详细,新手可以看得懂: http://www.360doc.com/content/16/0128/08/30422106_531162539.shtml

  2. HDU 4734 F(x)

    这题可能非递归版好写? #include<iostream> #include<cstdio> #include<cstring> #include<algo ...

  3. thinkphp中curl的使用,常用于接口

    /lib/action/PublicAction.class.php class PublicAction extends Action{ //curl,返回数组 public function ge ...

  4. 粗谈Android中的对齐

    在谈这个之前先啰嗦几个概念. 基线:书写英语单词时为了规范书写会设有四条线,从上至下第三条就是基线.基线对齐主要是为了两个控件中显示的英文单词的基线对齐,如下所示: Start:在看API的时候经常会 ...

  5. 数据结构(脑洞题,BIT):COGS 2394. 比赛

    比赛 时间限制:1 s   内存限制:256 MB [题目描述] n(n≤100000)个人编号为0到n-1,每人都有一个能力值,大小在0到n-1之间,各不相同,他们之间有c场比赛,每场比赛指定一个区 ...

  6. Redis安装介绍

    Redis安装介绍 一.Linux版本及配置 1.  Linux版本:Red Hat Enterprise Linux 6虚拟机 2.  配置: 内存:1G:CPU:1核:硬盘:20G 二.Redis ...

  7. 【.NETCore开源】开弓没有回头箭

    2019.2.11 开工大吉!经过了半个月的休假,今天回归岗位重新拾起工作,却发现熟悉的代码生疏了.年前的计划回忆不起来了,俗称"节后综合症". 忆半月圈子 过年放假的前几天有多篇 ...

  8. PyCharm Debug 调试

    断点(breakpoint),表示标记一行的位置,当程序运行到该行代码的时候,会将程序暂时暂停,以便对该行代码进行分析. 编辑python脚本,debug.py def hello(): return ...

  9. kafka 学习笔记

    一.为什么需要消息系统 1.解耦: 允许你独立的扩展或修改两边的处理过程,只要确保它们遵守同样的接口约束. 2.冗余: 消息队列把数据进行持久化直到它们已经被完全处理,通过这一方式规避了数据丢失风险. ...

  10. Spark笔记之DataFrameNaFunctions

    DataFrameNaFunctions用来对DataFrame中值为null或NaN的列做处理,处理分为三种类型: drop:根据条件丢弃含有null或NaN的行 fill:根据条件使用指定值填充值 ...