捕获ClientDataSet.ApplyUpdates和SocketConnection异常

核心提示:如何捕获ClientDataSet.ApplyUpdates的错误,不用ReconcileError...
var
  cdsEmp:TClientDataSet;
//保存
procedure  TfrmEmp.btnSave(Sender:  TObject);
begin
  cdsEmp.RemoteServer.AppServer.BegTrans;
  try
    cdsEmp.ApplyUpdates(0);  //更新错误在这一句发生,但是我却永远也捕获不到,
                             //我想自已在异常处理里显示这里发生的错误信息该怎么办?
    cdsQrObj.RemoteServer.AppServer.ComTrans;
  except
    on  E:Exception  do
    begin
      cdsEmp.RemoteServer.AppServer.RobTrans;
      Application.MessageBox(pchar('存盘失败!'+#13#10+'错误信息:'+E.Message),'提示',MB_OK+MB_ICONEXCLAMATION);
      Abort;
    end;
  end;
end;
//如果用这个错误处理,我的事务回滚却不知放在何处才妙,并且我不是想用这个错误处理
procedure  TfrmEmp.cdsEmpReconcileError(
   DataSet:  TCustomClientDataSet;  E:  EReconcileError;
   UpdateKind:  TUpdateKind;  var  Action:  TReconcileAction);
begin
   HandleReconcileError(DataSet,  UpdateKind,  E);
   Action:=raAbort;
end;
解答一:
//---------------------------------------------------------------
//据我所知,只能用ReconcileError 可以用下面的方法判断是否错误
...
BeginTransaction;
if cdsMaster.ApplyUpdates(0)+cdsDetail.ApplyUpdates(0)=0  then
  CommitTransaction
else
  RollbackTransaction;
ApplyUpdates方法返回更新时遇到的错误数量.
...
//---------------------------------------------------------------
 
解答二:
//---------------------------------------------------------------
在DataSetProvider的onUpdateError
raise  E;
然后就可以在客户端的
try
  ClientDataSet1.ApplyUpdates(0);
except
  on  e:Exception  do
  ...
end;
//----------------------------------------------------------------
解答三:
    其实真正的捕获ClientDataSet.ApplyUpdate异常的方法应该是在Apllication的异常中捕获并处理它。因为ClientDataSet抛出的异常为线程(进程?)异常,在ClientDataSet的ApplyUpdate中用try...except...end;是无法捕获的。
    具体方法为:在公共单元如DataModule中放置一个ApplicationEvent件,在该控件的OnException事件中捕获异常,该窗体应在所有有可能产生ApplyUpdate或Connection异常的窗体之前创建。
procedure TClient_RDataForm.ApplicationEvents1Exception(Sender: TObject;
  E: Exception);
begin
  if (E is ESocketConnectionError) or (E is ESocketError) then
  begin
    if not Is_OK then
    begin
      Application.MessageBox(PChar('考试应用服务器或网络连接失败!请退出后重新启动考试系统!  '),
                      '服务器连接中断', MB_OK + MB_ICONERROR);
      Application.Terminate;
      Exit;
    end;
    while not ReConnect_Srv do //重新连接又失败了
    begin
      if Application.MessageBox(PChar('考试服务器或网络连接失败!请立即与监考老师联系!  '+#13+'要重新搜索服务器请按[是],强制退出请按[否]!  '),
                      '服务器连接失败', MB_YESNO + MB_ICONSTOP) <> IDYES then
        if (Application.MessageBox('真的要强制退出考试系统吗?  ','强制退出确认', MB_YESNO + MB_ICONWARNING + MB_DEFBUTTON2) = IDYES) then
        begin
           Application.Terminate;
           Exit;
        end;
    end;
    if MyConnection.Connected then
    begin
      Application.MessageBox('考试应用服务器恢复连接成功!','连接成功',MB_OK+MB_ICONINFORMATION);
      Exit;
    end;
  end else
    raise Exception.Create('考试系统发生异常错误!退出后请重新启动考试系统继续考试!');
    //ShowMessage(e.Message);
end;




  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
TApplication.OnIdle   当应用程序变为空闲时,OnIdle事件发生。   使用OnIdle事件,可以写一个应用程序空闲时执行特定处理的事件处理程序。当应用程序不处理代码时,称为应用程序空闲。例如,当应用程序等待来自用户的输入时,应用程序为空闲。   TIdleEvent类型是OnIdle事件的类型,它指向一个应用程序空闲时运行的方法。TIdleEvent类型有一个布尔型参数Done,默认时该参数为True。若参数Done为True,当OnIdle事件返回时,调用Windows API WaitMessage函数。只有在应用程序消息队列中出现一个新消息时,WaitMessage函数才放弃对其他应用程序的控制。参数Done为False时,即使应用程序不忙,也不放弃对其他应用程序的控制。   当应用程序转移到空闲状态时,只调用一次OnIdle事件。除非参数Done设置为False,否则不连续调用OnIdle事件。将参数Done设置为False的应用程序,将消耗过多的CPU时间,从而影响整个系统性能。 在delphi中, 当在一个窗口上放置一个ApplicationEvents控件时,Application将会把所有的事件都转寄到ApplicationEvents; 也就是说,ApplicationEvents可以拦截到应用程序的全部事件,包括OnActivate\OnHelp\OnIdle\OnRestore\OnShortCut等等, 甚至可能通过OnMessage事件,在其中截取所有post到应用程序中所有窗口的消息,如WM_PAINT,WM_KEYDOWN, WM_KEYUP等常见的windows消息; 所以当有消息到来的时候就会触发它的OnMessage事件,在OnMessage中监视消息就可以了。 Action的事件有OnExecute和OnUpdate,OnExecute事件在控制被触发时响应,比如说按钮被按下,菜单被按下,而OnUpdate事件是在应用程序空闲时被调用, APPLICATIONEVENTS是用来捕获程序级事件的 ApplicationEvents1Message(var Msg: tagMSG;var Handled: Boolean); {通过 Perform 向窗体发送 消息; OnMessage 收不到} {通过 SendMessage 向窗体发送 消息; OnMessage 收不到} {通过 PostMessage 向窗体发送  消息; OnMessage 可以收到}

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值