关于 Indy 中 TIdTCPServer 在关闭时导致程序死机的Bug修复(delphi)

本文详细介绍了在使用Indy的TIdTCPServer控件时遇到的一个Bug,即在关闭服务时由于线程清理不完全导致的程序死机现象。通过对IdScheduler.pas单元文件的修改,成功解决了该问题,确保了服务能够正常关闭。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

目录

 

一、bug症状

二、Bug存在的版本

三、Bug解决方案


一、bug症状

在使用indy的TIdTCPServer控件时,我们打开服务时会设置控件属性Active为True,关闭服务时自然是设置控件属性Active为False。当我们打开TIdTCPServer服务,如果有一些客户端已经连接上,此时如果我们需要关闭TCP服务时,自然会设置控件属性Active为False,,大概率会导致程序死机,非常恼人。此时需要手工关闭进程才可以重新启动。这样使用起来就很不方便,当然我们也可以通过别的办法回避(我们先一个一个线程关闭,然后再让Active设置为False就可以回避这个问题)。导致这个问题的原因是有几个客户端连接,就会有几个对应的服务线程服务客户端,此时TIdTCPServer在设置Active为False的时候并没有清除干净这些线程,线程列表清除有Bug,就会导致程序死机。

二、Bug存在的版本

我修复的是下面的版本,其它版本是否存在问题不得而知,同学们可以根据这个版本对比自己的版本进行修复!

修复后文件下载连接:IdScheduler.pas

delphi版本:

Indy版本:

三、Bug解决方案

查找indy的源程序文件IdScheduler.pas单元,在这个文件单元中,对文件进行修改,修改前和修改后的代码如下:

修改前代码(为了看清楚,我是用的是截图):

procedure TIdScheduler.TerminateAllYarns;
var
  i: Integer;
  LList: TIdYarnList;
begin
  Assert(FActiveYarns<>nil);

  while True do begin
    // Must unlock each time to allow yarns that are terminating to remove themselves from the list
    LList := FActiveYarns.LockList;
    try
      if LList.Count = 0 then begin
        Break;
      end;
      for i := LList.Count - 1 downto 0 do begin
        TerminateYarn(
          {$IFDEF HAS_GENERICS_TList}LList.Items[i]{$ELSE}TIdYarn(LList.Items[i]){$ENDIF}
        );
      end;
    finally
      FActiveYarns.UnlockList;
    end;
                                            
    IndySleep(500); // Wait a bit before looping to prevent thrashing
  end;
end;

 

修改后代码:

procedure TIdScheduler.TerminateAllYarns;
var
  i: Integer;
  LList: TIdYarnList;
begin
  Assert(FActiveYarns<>nil);

  while True do begin
    // Must unlock each time to allow yarns that are terminating to remove themselves from the list
    LList := FActiveYarns.LockList;
    try
      if LList.Count = 0 then begin
        Break;
      end;
      for i := LList.Count - 1 downto 0 do begin
        TerminateYarn(
          {$IFDEF HAS_GENERICS_TList}LList.Items[i]{$ELSE}TIdYarn(LList.Items[i]){$ENDIF}
        );
        //sensor 2019-06-12
        LList.Delete(i);   //增加了这一句,因为上面循环使用的是 downto 所以可以直接删除!!!!
      end;
    finally
      FActiveYarns.UnlockList;
    end;

    IndySleep(500); // Wait a bit before looping to prevent thrashing
  end;
end;

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

海纳老吴

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值