首 页 网络编程
网页制作 图形图象 操作系统 冲浪宝典
软件教学 认证考试

网络安全 网络办公 行业资讯 评测对比
您当前位置:站长天空 -> 网络编程-> Visual Basic教程
用delphi模拟组件的两阶段提交_delphi教程
作者:网友供稿 点击:0
推荐
西部数码-全国虚拟主机10强!20余项虚拟主机管理功能,全国领先!第6代双线路虚拟主机,南北访问畅通无阻!可在线rar解压,自动数据恢复设置虚拟目录等.免费赠送访问统计,企业邮局.Cn域名注册10元/年,自助建站480元起,免费试用7天,满意再付款!P4主机租用799元/月.月付免压金
站内搜索
文章页数:[1] 
问题提出:写了一个数据库操作的类TDBOperate_DL,统一对数据库的操作嘛。提供了声明事务开始、提交事务和回滚事务的方法供其他类调用。
TDBOperate_DL = class
  private
    ADOC:TADOConnection;
    ADOQ:TADOQuery;
    isDestroyADOC:Boolean;    //是否销毁自己的ADOC?
    fIsInTrans:Boolean;        //是否已经开始事务
  public
    isCommit:Boolean;         //是否要提交事务,缺省是真,如果有类投票说反对提交,就为假
    function IsInTrans:Boolean;
    constructor Create(const newADOC:TADOConnection);overload;
    constructor Create(const ServerName,DataBaseName,UserID,Password:String);overload;
    destructor Destroy;override;
    procedure BeginTrans;
    procedure CommitTrans;
    procedure RollbackTrans;
    procedure Execute(const sqlString:String);
    function GetDataset(const sqlString:String):_Recordset;
    function GetConnection:TADOConnection;
    procedure SetConnection(const newADOC:TADOConnection);
  end;
       一些函数的实现:
procedure TDBOperate_DL.BeginTrans;          //开始事务
begin
  self.ADOC.BeginTrans;
  self.fIsInTrans := true;
end;
 
procedure TDBOperate_DL.CommitTrans;              //提交事务
begin
  self.ADOC.CommitTrans;
  self.fIsInTrans := false;
end;
 
procedure TDBOperate_DL.RollbackTrans;              //回滚事务
begin
  self.ADOC.RollbackTrans;
  self.fIsInTrans := false;
end;
 
function TDBOperate_DL.IsInTrans: Boolean;         //查看事务是否已开始
begin
  result := self.fIsInTrans;
end;
 
写了一个TThing类,用于向数据库中添加、修改或删除有关某种东西的记录,调用TDBOperate_DL类完成。为了调用方便,因此有关的事务就放在了TThing类中,外部调用时不用考虑事务了。
如:
procedure Tthing.Drop(const thing:String);
var
  sqlString:String;
begin
  sqlString := 删除的SQL语句;
  self.DBOperate.BeginTrans;           // DBOperate是TDBOperate_DL类型的私有变量,创建Tthing类实例时传入的参数。
  try
    self.DBOperate.Execute(sqlString);
    self.DBOperate.CommitTrans;
  except
    self. DBOperate.RollbackTrans;
    raise;
  end;
end;
后来又写了个TPerson类,用于向数据库中添加、修改或删除有关人的记录。同样事务放在了TPerson类中,现在我想删除人的记录时顺便调用TThing类删除和人有关的东西,事务问题就出现啦:事务不能嵌套啊。如果先删除TThing,再重新声明事务删除TPerson,如果TPerson出错,还怎么回滚TThing?
如:
procedure Tperson.Drop(const person:String);
var
  sqlString:String;
  thing:Tthing;
begin
  sqlString := 删除的SQL语句;
  thing := Tthing.Create(self.DBOperate);              //TDBOperate_DL类型的DBOperate是作为参数传进去的。
  Self.DBOperate.BeginTrans;
  Try
    Thing.Drop(person);               //里面有事务见上面的代码
    Self.DBOperate.Execute(sqlString);
    self.DBOperate.CommitTrans;
  except
    self.DBOperate.RollbackTrans;
    raise;
  end;
end;
解决方法,两阶段提交,先粘点背景知识:
不论两层或三层体系,事物处理都是通过两阶段提交实现的。在第一阶段,每个执行的的资源\记录被写入 事物环境(Transcation Context)中,然后资源协调者顺序查询每一个参与事物的执行是否成功,如果都没有问题的话,就进入第二阶段,每个执行都开始Commit它的操作。如果有一个执行有问题的话,资源协调者通知所有下属的执行放弃Commit,恢复数据原状态。
参考COM+的事务运行,如果一个组件是需要事务的,那么在组件创建时,事务就已经开始了,在组件销毁时进行事务投票,如果是根事务,则进行提交或回滚事务。(如果组件支持池化,这两种情况发生在组件激活和休眠事件中)。于是我们定义一个类如下。
//业务类的祖先类,用于提供统一的事务支持
  TTS_DL = class
  private
    isRootTrans:Boolean;      //是否是根事务
    isNeedTrans:Boolean;      //是否需要事务
  public
    DBOperate:TDBOperate_DL;   //操作数据库的类的实例
    procedure SetComplete;
    procedure SetAbort;
    constructor Create(const newDBOperate:TDBOperate_DL;needTrans:Boolean);//是否需要事务支持
    destructor Destroy;override;
  end;
在该类创建时,除了传递进操作数据库的类的实例外,再传入一个是否需要事务的标志,因为如果是只做读取数据库的操作,就用不着事务了。
类实现代码如下:
constructor TTS_DL.Create(const newDBOperate: TDBOperate_DL;
  needTrans: Boolean);
begin
  inherited Create;
  self.DBOperate := newDBOperate;             
  self.isNeedTrans := needTrans;                     //赋值是否需要事务
  if self.isNeedTrans then
  begin
    //如果在事务里,就不是根事务,保留事务上下文里isCommit的值不变
    if self.DBOperate.isInTrans then
      self.isRootTrans := false
    else
    begin
      self.DBOperate.BeginTrans;           //是根事务,就开始事务
      self.isRootTrans := true;
      self.DBOperate.isCommit := true;       //初始化提交标志为要提交事务
    end;
  end;
end;
 
destructor TTS_DL.Destroy;
begin
  if self.isNeedTrans then
  begin
    //如果是根事务,就按照投票结果进行事务提交或回滚
    if self.isRootTrans then
    begin
      if self.DBOperate.isCommit then
        self.DBOperate.CommitTrans
      else
        self.DBOperate.RollbackTrans;
    end;
  end;
  inherited;
end;
 
procedure TTS_DL.SetAbort;
begin
  self.DBOperate.isCommit := self.DBOperate.isCommit And false;       //投票说要回滚
end;
 
procedure TTS_DL.SetComplete;
begin
  self.DBOperate.isCommit := self.DBOperate.isCommit And true; //投票说要提交
end;
 
回到刚才的业务类Tthing和Tperson,这次都从TTS_DL类继承下来。
       Tthing = class(TTS_DL);
       Tperson = class(TTS_DL);
Tthing的删除代码该为如下:
procedure Tthing.Drop(const thing:String);
var
  sqlString:String;
begin
  sqlString := 删除的SQL语句;
  try
    self. DBOperate.Execute(sqlString);
    self. DBOperate.SetComplete;         //投票提交
  except
    self. DBOperate.SetAbort;                     //投票回滚
    raise;
  end;
end;
Tperson的删除代码如下:
procedure Tperson.Drop(const person:String);
var
  sqlString:String;
  thing:Tthing;
begin
  sqlString := 删除的SQL语句;
  thing := Tthing.Create(self. DBOperate,true);              //TDBOperate_DL类型的DBOperate是作为参数传进去的,true代表需要事务。
  Try
    Try
      Thing.Drop(person);
      Self.DBOperate.Execute(sqlString);
      self.DBOperate.SetComplete;           //投票提交
    except
      self. DBOperate.SetAbort;                       //投票回滚
      raise;
    end;
  finally
    thing.free;                                                           //记着一定要释放
  end;
end;
记着保持程序中使用唯一的操作数据库类TDBOperate_DL的实例,记着释放业务类实例,如果是需要事务的,尽量早释放,OK,搞定。
第一个版本,水平有限,还需要在实际应用中完善,全当抛砖引玉,请有经验的大侠们拍砖吧:)

文章整理:站长天空 网址:http://www.z6688.com/
以上信息与文章正文是不可分割的一部分,如果您要转载本文章,请保留以上信息,谢谢!

文章页数:[1] 


放大字体显示 缩小字体显示 打印文章 推荐给朋友
热门文章
·ASP.net中动态加载控件时一些问题的总结-ASP教程,ASP应用
·C#读写注册表-.NET教程,C#语言
·Visual Basic .NET中的异常处理简介(下)-.NET教程,VB.Net语言
·C#:文件的按行读/写及文件目录对话框的使用-.NET教程,C#语言
·ADO.Net:使用DataReader向数据库中插入数据-ASP教程,数据库相关
·列一张网恋赔偿清单(爆笑)
·如何用Photoshop画服装款式图-网页设计,Photoshop
·.NET下使用DataAdapter保存数据时,如何生成command语句及使用事务-.NET教程,数据库应用
·新型dc/dc电源控制芯片dpa426的应用
·ASP.NET 2.0 - Enter Key - Default Submit Button-.NET教程,Asp.Net开发
最新文章
·个人站长的网络赚钱两条新出路_网赚技巧
·adsense帐户最佳化纵深谈-adsense资深专员_网赚技巧
·google adsense容易被k的可能性列表_网赚技巧
·如何让程序被站长接受和产生利润_站长访谈
·马云,即成的中国互联网第4代霸主_站长访谈
·google关键词广告创建的十二招_google推广
·如何使google更快速收录你的新站_google推广
·几个颇有创意的网站推广方法_站长心得
·网络编辑:标题,如何让网民一见钟情(2)_网络编辑
·网站建设基础seo搜索引擎优化_seo网站优化
相关主题
  • 用delphi实现文件下载的几种方法_delphi教程
  • 用delphi创建服务程序_delphi教程
  • 用delphi做一个有颜色属性的按钮_delphi教程
  • 用delphi开发dll来代替8581协议控制和采集华为psm—a10电源(二)_delphi教程
  • 用delphi编写系统进程监控程序_delphi教程
  • 西部数码虚拟主机

    友情链接
    CNNIC 西部数码
    万网 自助建站
    虚拟主机 asp空间
    域名注册 域名
    域名申请 主页空间
    论坛空间 网站空间
    国际域名 虚拟空间
    空间租用 DDOS防火墙
    成都主机托管 四川主机托管
    主机租用 服务器租用
    网站目录 自助建站
    虚拟主机 网址大全
    软件下载
    自助链接
    虚拟主机资讯 特价虚拟主机
    版权申明:本站文章均来自网络,如有侵权,请联系我们,我们收到后立即删除,谢谢!
    关于我们:站长天空:专业提供最新的站长资讯、在线教程、虚拟主机权威评测、虚拟主机性能对比、网站制作教程,开发教程,站长工具。包括网页制作教程、冲浪宝典、编程参考、操作系统、软件教学、行业动态等。
    特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有。
    发表评论 打印  刷新     关闭