`
tiandirensoon
  • 浏览: 591871 次
文章分类
社区版块
存档分类
最新评论

在C#中使用COM+实现事务控制

 
阅读更多
.NET技术是微软大力推广的下一代平台技术,自从.NET技术架构的正式发布,此项技术也逐渐走向成熟和稳定。按照微软的平台系统占有率,我们不难想象得到,在未来的一两年内.NET技术必定会势如破竹一般的登上主流的技术平台,而一个新的技术平台得以快速发展的最重要的前提是:他不会彻底的摒弃以前的技术,这一点对于.NET技术来说指的就是COM/COM+技术了。

  一般来说,在IT技术界以及硬件产业,技术的更新换代速度非常得惊人,而惯例是所有的新技术都会遵循向下兼容的原则,但是.NET技术不仅仅做到了这一点,.NET甚至实现了相互之间的各自调用,这一点是非常难能可贵的。也就是说,不但我们可以在.NET组件中调用COM组件,同时也可以在COM组件中正常的调用.NET组件。这点带来的好处是显而易见的,一方面我们可以保持现有的技术资源,另一方面,在现有资源中可以利用.NET所带来的各种新技术。

  一般的数据库事务控制要求事务里所做的操作必须在同一个数据库内,这样在出现错误的时候才能回滚(RllBack)到初始状态。这就存在一个问题,在分布式应用程序中,我们往往需要同时操作多个数据库,使用数据库本身的事务处理,很难满足程序对事务控制的要求。在COM+中,提供了完整的事务服务,我们可以利用它来完成在分布式应用程序中的事务控制。

  具体过程如下

  一:用VS.NET生成一个类库。

  二:添加对System.EnterpristServices的引用,具体步骤

  菜单:(项目-添加引用-在.NET选项卡选择System.EnterpristServices-确定)

  三:构建类

  1:源程序

usingSystem;
usingSystem.EnterpriseServices;
usingSystem.Data.SqlClient;
usingSystem.Reflection;

namespaceCOMPlusSamples
{
//表明需要事务支持
[Transaction(TransactionOption.Required)]
//声明为服务器应用程序,还可以选择Library,表示为库应用程序
[assembly:ApplicationActivation(ActivationOption.Server)]
//描述信息
[assembly:Description("sample")]

publicclassTxCfgClass:ServicedComponent
{
privatestaticstringinit1="userid=sa;password=;initialcatalog=pubs;datasource=(local)";

privatestaticstringinit2="userid=sa;password=;initialcatalog=NorthWind;datasource=(local)";

privatestaticstringadd1="insertintoauthors('au_lname','au_fname')values('test1','test2')";

privatestaticstringadd2="insertintosamplevalues('test1',22)";
//theerrorsqlstatement
//thereisnottable“sample”

publicTxCfgClass(){}

privatevoidExecSQL(stringinit,stringsql)
{
SqlConnectionconn=newSqlConnection(init);
SqlCommandcmd=conn.CreateCommand();
cmd.CommandText=sql;
conn.Open();
cmd.ExecuteNonQuery();
conn.Close();
}

//添加一条记录到数据库
publicvoidAdd()
{
try
{
//在一数据库中插入一条记录

ExecSQL(init1,add1);
Console.WriteLine("theoperationinthesamedatabasecompletely");

//在另外一个数据库中插入两条记录
//这次执行的是一个错误的SQL语句

ExecSQL(init2,add2);
Console.WriteLine("theoperationintheotherdatabase
completely");

Console.WriteLine("Record(s)added,pressenter...");
Console.Read();

}
catch(Exceptione)
{
//事务回滚
ContextUtil.SetAbort();
Console.WriteLine("Becausetherearesomeerrorsintheoperation,sotranscationabort");
Console.WriteLine("Theerroris"+e.Message);
Console.WriteLine("abortsuccessfully");
Console.Read();
}
}
}
}
  2:程序说明:

  添加命名空间usingSystem.EnterpriseServices;因为本程序使用了其中的ContextUtil类
[Transaction(TransactionOption.Required)]说明DLL需要事务支持。

  本程序的TxCfgClass类从ServicedComponent类中继承,这样并不会影响该类,而只是在该类中添加了两个额外的方法,这两个方法可以使代码共享变得更加容易。

  程序使用的sqlserver数据库在本机运行,init1和init2是两个连接数据库的连接字符串,init连接pubs数据库,inin2连接northwind数据库,这是sql2000中自带的示例数据库。add1和add2是两条sql语句,作用是分别向两个数据库的表里添加一条记录。注意:add2是一条错误的语句,因为根本没有sample表,这样,会在执行时引起异常。(这正是我们所期望的)

  在执行到add2语句时,由于它是错误的,所以会引发异常,转到错误处理语句里来执行。

  ContextUtil.SetAbort();该语句使所有的数据库操作回滚,这样add1语句所插入的记录也将不存在。(达到预期目标)

  四:给程序添加强名(strongname)

  1:创建一对密钥

  用来创建密钥的工具是称为sn.exe的共享工具。通常通过命令提示运行它,该工具可执行各种任务以生成并提取密钥。我们需要用以下方式来运行sn.exe。

sn–kkey.snk

  其中key.snk代表将保存密钥的文件的名称。它的名称可以是任意的,不过习惯上带有.snk后缀名。

  2:签名

  签名通常是在编译时进行的。签名时,用户可利用C#属性通知编译器应该使用正确的密钥文件对DLL进行签名。要做到这一点用户需要打开工程中的AssemblyInfo.cs文件并进行修改。

  [assembly:AssemblyKeyFile(“..//..//key.snk”)]

  注:key.snk文件和项目文件在同一个文件夹

  五:编译成DLL(具体步骤)

  菜单:(生成-生成)

  如果一切正常,就会生成DLL文件

  六:使用regsvcs.exe将Dll注册到COM+Services里面

  我们需要用以下方式运行regsvcs.exe

  regsvcsdll文件名

  如果一切正常的话,regsvcs.exe就会把dll输入到COM+Services中。

  至此,我们已经生成并注册了这个可以由其它程序使用的类,现在,我们来写一个控制台程序来检验这个类是否正常运行

  七:构建客户机

  1:新建控制台应用程序项目

  菜单(文件-新建-项目)

  选择控制台应用程序,并选择添入解决方案,确定

  2:同上面的第二步一样,添加对System.EnterpriseServices的引用。

  3:添加对自己刚才做好的类的引用。

  菜单(项目-添加引用-浏览),选择刚才生成的DLL,确定

  4:输入以下程序

usingSystem;
usingCOMPlusSamples;
usingSystem.EnterpriseServices;

publicclassClient
{
publicstaticvoidMain()
{
TxCfgClasscfg=newTxCfgClass();
cfg.Add();
}
}

  5:将控制台程序设置为启动项,然后编译运行,就会看到结果。

  正如我们希望的,第一条记录没有插入数据库
分享到:
评论

相关推荐

    C#COM+分布式事务

    本例子讲述了C# 分布式事务的完整使用.

    用VC#实现分布式COM+服务之实现数据库分布事务

    COM+分布式事务处理

    C#开发基于FreeSql多库分布式事务、跨库查询、跨库分页查询、跨库增删改等功能实现源码+项目说明+sln.zip

    既然分库了 分布式事务怎么处理,说到分布式事务 常见的解决方案有TCC/SAGA/消息队列最终一致性,在.NET生态中有基于消息队列实现的分布式事务 [CAP](https://github.com/dotnetcore/CAP) ,TCC和SAGA调研了很久没有...

    C#开发经验技巧宝典

    0930 在查询程序中使用变量 544 0931 对查询结果进行排序 544 0932 批量获取结果集信息 545 0933 对查询结果生成表 545 0934 实现数据类型转换 546 0935 获取当前数据库的登录用户名 546 0936 如何正确...

    C#编程经验技巧宝典

    92 <br>0140 如何对计算结果四舍五放入 92 <br>0141 如何将商品金额小写转换成大写 92 <br>0142 如何根据生日自动计算员工年龄 93 <br>0143 如何设置货币值中使用的小数位数 93 <br>0144 ...

    C#微软培训资料

    C#语言在.NET 框架中的作用及其特性 1.1 Microsoft.NET 一场新的革命 1.1.1 什么是.NET 2000 年 6 月 22 日 不论对 Microsoft 还是对整个 IT 业界都将成为值得纪念的一天 这一天 微软公司正式推出...

    C#与.NET3.5高级程序设计(第4版) 中文4

    24.7 在内存文档中导航 680 24.8 小结 682 第25章 WCF 683 25.1 各种分布式计算API 683 25.2 WCF的作用 688 25.3 WCF核心程序集 690 25.4 Visual Studio WCF项目模板 691 25.5 WCF应用程序的基本构成...

    asp.net知识库

    在 SQL Server 2005 中使用表值函数来实现空间数据库 SQL Server 2005的30个最重要特点 同时安装sql2000和sql2005的经验 类如何与界面绑定 在Asp.net中如何用SQLDMO来获取SQL Server中的对象信息 使用Relations建立...

    C#与.NET技术平台实战演练.part1

    initializer10-5初始化只读数据10-6在构造器中使用out与ref10-7struct构造器10-7-1struct构造器的限制10-8static构造器10-8-1使用static构造器初始化静态成员10-8-2static构造器的限制10-9对象与内存10-9-l对象的...

    ORM框架-VB/C#.Net实体代码生成工具(EntitysCodeGenerate)ECG4.3.pdf

    2.2 在开发中的实际应用 5 2.2.1 单个实体对象的数据库操作 6 1、获取一个实体对象信息 6 2、插入一个实体对象信息 6 3、更新一个实体对象信息 6 4、保存一个实体对象信息 6 5、删除一个实体对象信息 7 6、取得实体...

Global site tag (gtag.js) - Google Analytics