COM+时代的自动事务
最近看公司的遗留项目代码,调试的时候发现经常报分布式事务错误,可是整个代码里没有看见开启过事务,于是开始研究,发现了这个.Net Framework1.1时代的产物。
namespace BusinessService
{
[Transaction(TransactionOption.RequiresNew, Isolation = TransactionIsolationLevel.ReadCommitted)]
public class Operation : ServicedComponent, IOperation
//public class Operation : MarshalByRefObject, IOperation
{
private static List<Person> persons = new List<Person>();
private static string dbStr = "Data Source=DESKTOP-PABLR9F;Initial Catalog=MyDB;Integrated Security=True;Connect Timeout=15;Encrypt=False;TrustServerCertificate=True;ApplicationIntent=ReadWrite;MultiSubnetFailover=False";
public Operation()
{
Console.WriteLine("服务激活...");
}
public int AddPerson(Person person)
{
Console.WriteLine("添加Person...");
var s = persons.FirstOrDefault(x => x.Id == person.Id);
if(s==null)
{
persons.Add(person);
;
}
else
{
;
}
}
[AutoComplete(true)]
public void AddPersons()
{
List<Person> list = new List<Person>();
; i <= ; i++)
{
Person p = + i };
list.Add(p);
}
using (SqlConnection conn = new SqlConnection(dbStr))
{
if (conn.State != System.Data.ConnectionState.Open)
conn.Open();
using (SqlCommand cmd = conn.CreateCommand())
{
foreach (var item in list)
{
string sql = string.Format("insert into student(name,age,lasttime) values( '{0}',{1},'{2}')",item.Name,item.Age,DateTime.Now);
cmd.CommandText = sql;
cmd.ExecuteNonQuery();
}
}
}
}
[AutoComplete(true)]
public void AddPersonRange(ArrayList list)
{
using (SqlConnection conn = new SqlConnection(dbStr))
{
if (conn.State != System.Data.ConnectionState.Open)
conn.Open();
using (SqlCommand cmd = conn.CreateCommand())
{
foreach (Person item in list)
{
string sql = string.Format("insert into student(name,age,lasttime) values( '{0}',{1},'{2}')", item.Name, item.Age, DateTime.Now);
cmd.CommandText = sql;
cmd.ExecuteNonQuery();
}
}
}
}
public Person GetPerson(int id)
{
Console.WriteLine("获取Person...");
return persons.FirstOrDefault(x => x.Id == id);
}
public DateTime GetTime()
{
Console.WriteLine("获取时间:"+DateTime.Now);
return DateTime.Now;
}
}
}
让类继承ServicedComponent,这样调用类的方法的时候会自动开启事务
然后在方法上加上[AutoComplete(true)]属性,这样在方法结束的时候会自动结束事务
这个项目是用remoting做的,顺便把这个也看了看
整体框架:

接口层:
namespace BusinessContract
{
[Serializable]
public class Person
{
public int Id { get; set; }
public string Name { get; set; }
public int Age { get; set; }
public DateTime LastTime { get; set; }
}
}
服务层:
namespace BusinessService
{
[Transaction(TransactionOption.RequiresNew, Isolation = TransactionIsolationLevel.ReadCommitted)]
public class Operation : ServicedComponent, IOperation
//public class Operation : MarshalByRefObject, IOperation
{
private static List<Person> persons = new List<Person>();
private static string dbStr = "Data Source=DESKTOP-PABLR9F;Initial Catalog=MyDB;Integrated Security=True;Connect Timeout=15;Encrypt=False;TrustServerCertificate=True;ApplicationIntent=ReadWrite;MultiSubnetFailover=False";
public Operation()
{
Console.WriteLine("服务激活...");
}
public int AddPerson(Person person)
{
Console.WriteLine("添加Person...");
var s = persons.FirstOrDefault(x => x.Id == person.Id);
if(s==null)
{
persons.Add(person);
;
}
else
{
;
}
}
[AutoComplete(true)]
public void AddPersons()
{
List<Person> list = new List<Person>();
; i <= ; i++)
{
Person p = + i };
list.Add(p);
}
using (SqlConnection conn = new SqlConnection(dbStr))
{
if (conn.State != System.Data.ConnectionState.Open)
conn.Open();
using (SqlCommand cmd = conn.CreateCommand())
{
foreach (var item in list)
{
string sql = string.Format("insert into student(name,age,lasttime) values( '{0}',{1},'{2}')",item.Name,item.Age,DateTime.Now);
cmd.CommandText = sql;
cmd.ExecuteNonQuery();
}
}
}
}
[AutoComplete(true)]
public void AddPersonRange(ArrayList list)
{
using (SqlConnection conn = new SqlConnection(dbStr))
{
if (conn.State != System.Data.ConnectionState.Open)
conn.Open();
using (SqlCommand cmd = conn.CreateCommand())
{
foreach (Person item in list)
{
string sql = string.Format("insert into student(name,age,lasttime) values( '{0}',{1},'{2}')", item.Name, item.Age, DateTime.Now);
cmd.CommandText = sql;
cmd.ExecuteNonQuery();
}
}
}
}
public Person GetPerson(int id)
{
Console.WriteLine("获取Person...");
return persons.FirstOrDefault(x => x.Id == id);
}
public DateTime GetTime()
{
Console.WriteLine("获取时间:"+DateTime.Now);
return DateTime.Now;
}
}
}
服务端:
static void Main(string[] args)
{
//IClientChannelSinkProvider clientProvider = new BinaryClientFormatterSinkProvider();
//IServerChannelSinkProvider serverProvider = new BinaryServerFormatterSinkProvider();
//IDictionary dict = new Hashtable();
//dict["port"] = 8080;
//dict["name"] = "myServer";
//HttpChannel channel = new HttpChannel(dict, clientProvider, serverProvider);
//int port = Convert.ToInt32(ConfigurationManager.AppSettings["port"].ToString());
//HttpChannel channel = new HttpChannel(port);
//ChannelServices.RegisterChannel(channel, false);
//RemotingConfiguration.RegisterWellKnownServiceType(typeof(Operation), "ServiceOperation", WellKnownObjectMode.SingleCall);
RemotingConfiguration.Configure(AppDomain.CurrentDomain.SetupInformation.ConfigurationFile, false);
Console.WriteLine("服务端启动...");
Console.ReadKey();
}
服务端配置:
<system.runtime.remoting>
<!--.net remoting配置的根节点-->
<application>
<!--包含有关远程应用程序使用及公共的对象信息-->
<service>
<!--服务器端,用于指定以什么方式公开什么对象-->
<wellknown mode="Singleton" type="BusinessService.Operation,BusinessService" objectUri="ServiceOperation"/>
<!--与wellknown相对是activated,wellknown表示服务器端激活对象,后者为客户端激活对象-->
<!--mode表示如何响应客户端请求,Singleton表示单一实例-->
<!--type表示要公开的类型,选择指定类型名称(含名称空间),然后指定类型所属.dll文件-->
<!--objectUri表示远程对象访问路径-->
</service>
<channels>
<!--用于指定通道信息,可心同时指定多个通道-->
<channel port=" ref="http">
<clientProviders>
<formatter ref="binary" />
</clientProviders>
</channel>
<!--具体的通道信息,port为端口号,ref为引用的通道类型-->
<!--.net 框架提供了http与tcp通道-->
</channels>
</application>
</system.runtime.remoting>
客户端:
static void Main(string[] args)
{
//HttpChannel channel = new HttpChannel();
//ChannelServices.RegisterChannel(channel, false);
RemotingConfiguration.Configure(AppDomain.CurrentDomain.SetupInformation.ConfigurationFile, false);
string serverUrl = ConfigurationManager.AppSettings["serverUrl"].ToString();
IOperation op = (IOperation)Activator.GetObject(typeof(IOperation), serverUrl);
Console.WriteLine(op.GetTime());
//Console.ReadLine();
Person person = , Name = };
op.AddPerson(person);
Person p = op.GetPerson();
Console.WriteLine(p.Name + ":" + p.Age);
List<Person> list = new List<Person>();
ArrayList al = new ArrayList();
;i<=;i++)
{
Person p1 = + i };
list.Add(p1);
al.Add(p1);
}
op.AddPersons();
op.AddPersonRange(al);
Console.WriteLine("Done");
Console.ReadKey();
}
客户端配置:
<system.runtime.remoting>
<application>
<client>
<wellknown url="http://10.200.80.172:8080/ServiceOperation" type="BusinessContract.IOperation,BusinessContract"/>
</client>
<!--远程对象访问路径,域名与IP地址都可以-->
<channels>
<channel ref="http"/>
<!--port为0表示客户端不侦听任何端口-->
<clientProviders>
<formatter ref="binary" />
</clientProviders>
<serverProviders>
<formatter ref="binary" />
</serverProviders>
</channels>
</application>
</system.runtime.remoting>
要注意的是:需要把程序集设置为[assembly: ComVisible(true)]

还需要对程序集进行签名:

COM+时代的自动事务的更多相关文章
- 利用AOP实现SqlSugar自动事务
先看一下效果,带接口层的三层架构: BL层: public class StudentBL : IStudentService { private ILogger mLogger; private r ...
- Hibernate自动事务揪出的编码不规范
最近重构的项目(Java初学中),Service层一个获取通知记录报错: org.springframework.dao.InvalidDataAccessResourceUsageException ...
- Aop实现SqlSugar自动事务
http://www.cnblogs.com/jaycewu/p/7733114.html
- 在 ASP.NET Core 中自动启用 CAP 事务
本篇文章旨在描述如何在 ASP.NET Core项目中并以一种简便的方式启用CAP事务,因为在我们的示例中都是直接演示比较直观的方式,没有进行封装,有些初学者同学不太会,找到问我如何封装,本篇文章主要 ...
- sql编程篇 (五) 事务
计算机中的事务 编辑 概念 事务(Transaction)是访问并可能更新数据库中各种数据项的一个程序执行单元(unit).事务通常由高级数据库操纵语言或编程语言(如SQL,C++或Java)书写的用 ...
- CI框架--事务
CI框架中事务封装的很完善,使用起来很简单 1.不开启事务 //不开启事务 前两个sql 能够执行成功,第三个执行失败 $this->device_model->addForCamera( ...
- hibernate(九) 二级缓存和事务级别详讲
序言 这算是hibernate的最后一篇文章了,下一系列会讲解Struts2的东西,然后说完Struts2,在到Spring,然后在写一个SSH如何整合的案例.之后就会在去讲SSM,在之后我自己的个人 ...
- SQLite事务管理
事务管理对数据库一致性是至关重要的.数据库实现ACID属性以确保一致性.SQLite依赖于本地文件锁和页日志来实现ACID属性.SQLite只支持扁平事务,并不支持事务嵌套和保存点能力. 1.1 事务 ...
- SqlServer_事务
事务处理是在数据处理时经常遇到的问题,经常用到的方法有以下三种总结整理如下:方法1:直接写入到sql 中在存储过程中使用 BEGIN TRANS, COMMIT TRANS, ROLLBACK TRA ...
随机推荐
- js判断 pc 手机 浏览器
<script> var result = window.matchMedia('(max-width: 700px)'); var browser={ versions:function ...
- 公司内部Samba 服务器架设
1.需求 在公司内部打造一个文件管理系统,其作用域仅仅在公司内部,支持在线对文件的修改和保存操作等,同时也要注意权限问题. 2.策划 目前设立四个群组:运维.开发 .测试和普通,当然所对应的对文件的访 ...
- 快速排序 JavaScript 实现
作为算法目录下的第一篇博文,快速排序那是再合适不过了.作为最基本最经典的算法之一,我觉得每个程序员都应该熟悉并且掌握它,而不是只会调用库函数,知其然而不知其所以然. 排序算法有10种左右(或许更多), ...
- java的静态内部类
只是一个简单的记录.因为一直排斥java这个东西.java跟c++比是很不错的一个语言,至少内存管理这么麻烦的东西不用操心了.但是和不断崛起的脚本语言比起来,效率差的太多.无论如何做android还是 ...
- 【python-selenium】python-selenium安装配置
selenium 是一个web的自动化测试工具,不少学习功能自动化的同学开始首选selenium ,相因为它相比QTP有诸多有点: * 免费,也不用再为破解QTP而大伤脑筋 * 小巧,对于不同的语 ...
- 修改Android EditText光标颜色
EditText有一个属性:android:textCursorDrawable,这个属性是用来控制光标颜色的 android:textCursorDrawable="@null&quo ...
- hdu 4135 [a,b]中n互质数个数+容斥
http://acm.hdu.edu.cn/showproblem.php?pid=4135 给定一个数n,求某个区间[a,b]内有多少数与这个数互质. 对于一个给定的区间,我们如果能够求出这个区间内 ...
- excel设定备选值
excel设定备选值 有的时候我们要人为向excel中某一列添加数据,可以通过下面的方法,为这列设定备选值. 操作方法 选中excel表格的一列,选择 数据 -- 有效性 -- 允许: 选择 序列 ...
- Swift3 页面顶部实现拉伸效果代码
//懒加载 //顶部需要拉伸自定义视图 lazy var headView:MyHeaderView = { //let hframe = CGRect(x: 0, y: 0, width: swid ...
- 在.net中使用ETW事件的方法
直到.net4.5,才有了比较便利的操作ETW的方法. 本文介绍的方法主要来源于Microsoft.Diagnostics.Tracing.TraceEvent官方资料库. 准备 (1)需要用到类:M ...