在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:源程序
using System;
using System.EnterpriseServices;
using System.Data.SqlClient;
using System.Reflection;
namespace COMPlusSamples
{
//表明需要事务支持
[ Transaction(TransactionOption.Required) ]
//声明为服务器应用程序,还可以选择Library,表示为库应用程序
[assembly: ApplicationActivation(ActivationOption.Server)]
//描述信息
[assembly: Description("sample")]
public class TxCfgClass : ServicedComponent
{
private static string init1 = "user id=sa;password=;initial catalog=pubs;data source=(local)";
private static string init2 = "user id=sa;password=;initial catalog=NorthWind;data source=(local)";
private static string add1 = "insert into authors('au_lname','au_fname') values('test1', 'test2')";
private static string add2 = "insert into sample values('test1',22)";
//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 = conn.CreateCommand();
cmd.CommandText = sql;
conn.Open();
cmd.ExecuteNonQuery();
conn.Close();
}
//添加一条记录到数据库
public void Add()
{
try
{
//在一数据库中插入一条记录
ExecSQL(init1, add1);
Console.WriteLine("the operation in the same database completely");
//在另外一个数据库中插入两条记录
//这次执行的是一个错误的SQL语句
ExecSQL(init2, add2);
Console.WriteLine("the operation in the other database
completely");
Console.WriteLine("Record(s) added, press enter...");
Console.Read();
}
catch(Exception e)
{
//事务回滚
ContextUtil.SetAbort();
Console.WriteLine("Because there are some errors in the operation ,so transcation abort");
Console.WriteLine("The error is " + e.Message);
Console.WriteLine("abort successfully");
Console.Read();
}
}
}
}
2:程序说明:
添加命名空间 using System.EnterpriseServices;因为本程序使用了其中的ContextUtil类
[ Transaction(TransactionOption.Required) ] 说明DLL需要事务支持。
本程序的TxCfgClass 类从ServicedComponent类中继承,这样并不会影响该类,而只是在该类中添加了两个额外的方法,这两个方法可以使代码共享变得更加容易。
程序使用的sql server数据库在本机运行,init1 和 init2是两个连接数据库的连接字符串,init连接pubs数据库,inin2连接northwind数据库,这是sql2000中自带的示例数据库。add1和add2是两条sql语句,作用是分别向两个数据库的表里添加一条记录。注意:add2是一条错误的语句,因为根本没有sample表,这样,会在执行时引起异常。(这正是我们所期望的)
在执行到add2语句时,由于它是错误的,所以会引发异常,转到错误处理语句里来执行。
ContextUtil.SetAbort();该语句使所有的数据库操作回滚,这样add1语句所插入的记录也将不存在。(达到预期目标)
四:给程序添加强名(strong name)
1:创建一对密钥
用来创建密钥的工具是称为sn.exe的共享工具。通常通过命令提示运行它,该工具可执行各种任务以生成并提取密钥。我们需要用以下方式来运行sn.exe。
sn –k key.snk
其中key.snk 代表将保存密钥的文件的名称。它的名称可以是任意的,不过习惯上带有.snk后缀名。
2:签名
签名通常是在编译时进行的。签名时,用户可利用C#属性通知编译器应该使用正确的密钥文件对DLL进行签名。要做到这一点用户需要打开工程中的AssemblyInfo.cs文件并进行修改。
[assembly:AssemblyKeyFile(“..//..//key.snk”)]
注:key.snk文件和项目文件在同一个文件夹
五:编译成DLL (具体步骤)
菜单:(生成-生成)
如果一切正常,就会生成DLL文件
六:使用regsvcs.exe将Dll注册到COM+ Services里面
我们需要用以下方式运行regsvcs.exe
regsvcs dll文件名
如果一切正常的话,regsvcs.exe就会把dll输入到COM+ Services中。
至此,我们已经生成并注册了这个可以由其它程序使用的类,现在,我们来写一个控制台程序来检验这个类是否正常运行
七:构建客户机
1:新建控制台应用程序项目
菜单(文件-新建-项目)
选择控制台应用程序 ,并选择 添入解决方案 ,确定
2:同上面的第二步一样,添加对System.EnterpriseServices的引用。
3:添加对自己刚才做好的类的引用。
菜单(项目-添加引用-浏览),选择刚才生成的DLL,确定
4:输入以下程序
using System;
using COMPlusSamples;
using System.EnterpriseServices;
public class Client
{
public static void Main()
{
TxCfgClass cfg = new TxCfgClass();
cfg.Add();
}
}
5:将控制台程序设置为启动项,然后编译运行,就会看到结果。
正如我们希望的,第一条记录没有插入数据库
在C#中使用COM+实现事务控制相关推荐
- Spring中的事务控制
Chapter 1. Spring中的事务控制(Transacion Management with Spring) Table of Contents 1.1. 有关事务(Transaction)的 ...
- Spring中的事务控制(Transacion Management with Spring)
1.1. 有关事务(Transaction)的楔子 1.1.1. 认识事务本身1.1.2. 初识事务家族成员 1.2. 群雄逐鹿下的Java事务管理 1.2.1. Java平台的局部事务支持1.2.2 ...
- Spring中的事务控制学习中(转)
1.1. 有关事务(Transaction)的楔子 1.1.1. 认识事务本身 1.1.2. 初识事务家族成员 1.2. 群雄逐鹿下的Java事务管理 1.2.1. Java平台的局部事务支持 1.2 ...
- Spring中的事务控制学习中
Chapter 1. Spring中的事务控制(Transacion Management with Spring) Table of Contents 事务管理(Transaction Manage ...
- php中对MYSQL操作之事务控制,回滚
<?php //事务控制,回滚 //创建一个mysqli对象 $mysqli = new MySQLi("主机名","mysql用户名","密码 ...
- 全局性事务控制如何在springboot中配置
开发中,我们一般会利用AOP配置全局性的事务,对指定包下指定的方法(如add,update等)进行事务控制,在springboot中如何实现呢? @EnableTransactionManagemen ...
- spring中事务控制的一组API
Spring事务控制我们要明确的 第一:JavaEE体系进行分层开发,事务处理位于业务层,Spring提供了分层设计业务层的事务处理解决方案. 第二:spring框架为我们提供了一组事务控制的接口.具 ...
- t-sql中的事务控制及错误处理
------------------------------------------------事务控制------------------------------------------------ ...
- 在Nutz中如何配置多个数据库源,并且带事务控制
在Nutz中如何配置多个数据库源,并且带事务控制 发布于 560天前 作者 Longitude 995 次浏览 复制 上一个帖子 下一个帖子 标签: 无 在Nutz中如何配置多个数据库源, ...
最新文章
- 8、Kubernetes核心技术Service
- Binary Gap(二进制空白)
- 《Leadership and the One Minute Manager》读书笔记之一
- 技术人写的公众号为啥没人看?
- 怎样做好域名防红_微信域名防拦截的工作
- datagrid getselected/getselections/getData之间的用法差异
- 批量保存打开的网页到本地
- pgAdmin III简介
- 目标检测算法——YOLOv3
- linux系统路由器地址查询,查找路由器登录IP地址指南
- 计算机应用研究参考文献格式,参考文献编写规则-计算机应用研究.PDF
- VMware Workstation的安装
- android botton控件基本属性
- Entity Framework Core系列教程-3为现有数据库生成实体模型
- telnet无法打开到主机的连接。 在端口 23: 连接失败
- 【WebApp】离线webapp (android)开发
- php语言程序设计总结,高校邦PHP语言程序设计答案
- 如何下载宝坻区卫星地图高清版大图
- Android 高德地图绘制线、添加图钉、添加线段纹理
- AndroidStudio之option menu菜单的使用,android游戏开发
热门文章
- java thread 无法执行_哪位大神帮我讲一下这段代码,为什么线程不能继续执行
- 一文看懂docker容器技术架构及其中的各个模块
- 使用Python和OpenCV检测图像中的条形码
- 工业界AI项目落地血泪教训总结
- 【opencv】(2) 图像处理:边界填充、图像融合、图像阈值、数值计算
- 解决Hbuilder打包的APP微信支付时无法唤起支付,且提示{“code“:-100,“message“:“[payment微信:-1]General errors“}的问题
- PHP函数printf()、sprintf()的用法
- 使用Blender中的几何节点创建程序对象
- linux+Qt 下利用D-Bus进行进程间高效通信的三种方式
- Linux命令行好玩的命令