c#

位置:IT落伍者 >> c# >> 浏览文章

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


发布日期:2021年05月23日
 
在C#中使用COM+实现事务控制

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

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

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

具体过程如下

用VSNET生成一个类库

添加对SystemEnterpristServices的引用具体步骤

菜单(项目-添加引用-在NET选项卡选择SystemEnterpristServices-确定)

构建类

源程序

using System;

using SystemEnterpriseServices;

using SystemDataSqlClient;

using SystemReflection;

namespace COMPlusSamples

{

//表明需要事务支持

[ Transaction(TransactionOptionRequired) ]

//声明为服务器应用程序还可以选择Library表示为库应用程序

[assembly: ApplicationActivation(ActivationOptionServer)]

//描述信息

[assembly: Description(sample)]

public class TxCfgClass : ServicedComponent

{

private static string init = user id=sa;password=;initial catalog=pubs;data source=(local);

private static string init = user id=sa;password=;initial catalog=NorthWind;data source=(local);

private static string add = insert into authors(au_lnameau_fname) values(test test);

private static string add = insert into sample values(test);

//the error sql statement

//there is not table sample

public TxCfgClass() {}

private void ExecSQL(string init string sql)

{

SqlConnection conn = new SqlConnection(init);

SqlCommand cmd = connCreateCommand();

cmdCommandText = sql;

connOpen();

cmdExecuteNonQuery();

connClose();

}

//添加一条记录到数据库

public void Add()

{

try

{

//在一数据库中插入一条记录

ExecSQL(init add);

ConsoleWriteLine(the operation in the same database completely);

//在另外一个数据库中插入两条记录

//这次执行的是一个错误的SQL语句

ExecSQL(init add);

ConsoleWriteLine(the operation in the other databasecompletely);

ConsoleWriteLine(Record(s) added press enter);

ConsoleRead();

}

catch(Exception e)

{

//事务回滚

ContextUtilSetAbort();

ConsoleWriteLine(Because there are some errors in the operation so transcation abort);

ConsoleWriteLine(The error is + eMessage);

ConsoleWriteLine(abort successfully);

ConsoleRead();

}

}

}

}

程序说明

添加命名空间 using SystemEnterpriseServices;因为本程序使用了其中的ContextUtil类

[ Transaction(TransactionOptionRequired) ] 说明DLL需要事务支持

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

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

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

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

给程序添加强名(strong name)

创建一对密钥

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

sn –k keysnk

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

签名

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

[assembly:AssemblyKeyFile(\\\\keysnk)]

keysnk文件和项目文件在同一个文件夹

编译成DLL (具体步骤)

菜单(生成-生成)

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

使用regsvcsexe将Dll注册到COM+ Services里面

我们需要用以下方式运行regsvcsexe

regsvcs dll文件名

如果一切正常的话regsvcsexe就会把dll输入到COM+ Services中

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

构建客户机

新建控制台应用程序项目

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

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

同上面的第二步一样添加对SystemEnterpriseServices的引用

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

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

输入以下程序

using System;

using COMPlusSamples;

using SystemEnterpriseServices;

public class Client

{

public static void Main()

{

TxCfgClass cfg = new TxCfgClass();

cfgAdd();

}

}

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

正如我们希望的第一条记录没有插入数据库

               

上一篇:ADO.NET和ADO的比较

下一篇:C#中构造函数和析构函数的用法(一)