5

我需要为TTcpClient. 我认为默认超时时间约为 20-25 秒,但我需要将其更改为 500 毫秒。有可能吗?如何?

procedure TForm1.Button1Click(Sender: TObject);
   begin
     TcpClient2.RemoteHost := '192.168.1.1';
     TcpClient2.RemotePort := '23';
     TcpClient2.Connect;

     tcpclient2.Receiveln();
     tcpclient2.Sendln('admin');
     tcpclient2.Receiveln;
   end;

我尝试non-blocking了选项,但单击按钮后软件返回错误,我必须再做 4-5 次。有什么帮助吗?

谢谢 :)

4

1 回答 1

8

Winsock 没有连接超时,但这可以克服。

你有几个选择:

  1. 没有线程:

    • 使用非阻塞模式:调用Connect,然后使用Winsock select函数等待(封装在TTcpClient 继承的TBaseSocket Select方法中)。

    • 使用阻塞模式:暂时更改为非阻塞模式,并像前一种情况一样继续。

  2. 使用线程:请参阅 Remy Lebeau 对如何使用 Winsock API 控制连接超时的回答?.

  3. 使用印地。

阻塞与非阻塞

使用阻塞或非阻塞模式是一个非常重要的设计决策,它将影响您的许多代码,并且您以后无法轻易更改。

例如,在非阻塞模式下,接收函数 (as Receiveln) 不会等到有足够的输入可用并且可以返回一个空字符串。如果这是您需要的,这可能是一个优势,但您需要实施一些策略,例如TcpClient.WaitForData在调用接收函数之前等待使用(在您的示例中,Receiveln-Sendln-Receiveln不会按原样工作)。

对于简单的任务,阻塞模式更容易处理。

非阻塞模式

以下函数将等待连接成功或超时:

function WaitUntilConnected(TcpClient: TTcpClient; Timeout: Integer): Boolean;
var
  writeReady, exceptFlag: Boolean;
begin
  // Select waits until connected or timeout
  TcpClient.Select(nil, @writeReady, @exceptFlag, Timeout);
  Result := writeReady and not exceptFlag;
end;

如何使用:

// TcpClient.BlockMode must be bmNonBlocking

TcpClient.Connect; // will return immediately
if WaitUntilConnected(TcpClient, 500) then begin // wait up to 500ms
  ... your code here ...
end;

还要注意 TTcpClient 的非阻塞模式设计中的以下缺点/缺陷:

  • 几个函数将调用OnError设置SocketErrorWSAEWOULDBLOCK(10035)。
  • Connected属性将是false因为分配在Connect.

阻塞模式

连接超时可以通过在创建套接字之后调用之前更改为非阻塞模式Connect,并在调用之后恢复为阻塞模式来实现。

这有点复杂,因为TTcpClient如果我们更改,则关闭连接和套接字BlockMode,并且也没有直接的方法来单独创建套接字而不连接它。

为了解决这个问题,我们需要在创建套接字之后但在连接之前进行挂钩。这可以使用DoCreateHandle受保护的方法或OnCreateHandle事件来完成。

最好的方法是从 TTcpClient 派生一个类并使用DoCreateHandle,但是如果出于任何原因需要直接使用 TTcpClient 而不使用派生类,则可以使用 轻松重写代码OnCreateHandle

type
  TExtendedTcpClient = class(TTcpClient)
  private
    FIsConnected: boolean;
    FNonBlockingModeRequested, FNonBlockingModeSuccess: boolean;
  protected
    procedure Open; override;
    procedure Close; override;
    procedure DoCreateHandle; override;
    function SetBlockModeWithoutClosing(Block: Boolean): Boolean;
    function WaitUntilConnected(Timeout: Integer): Boolean;
  public
    function ConnectWithTimeout(Timeout: Integer): Boolean;
    property IsConnected: boolean read FIsConnected;
  end;

procedure TExtendedTcpClient.Open;
begin
  try
    inherited;
  finally
    FNonBlockingModeRequested := false;
  end;
end;

procedure TExtendedTcpClient.DoCreateHandle;
begin
  inherited;
  // DoCreateHandle is called after WinSock.socket and before WinSock.connect
  if FNonBlockingModeRequested then
    FNonBlockingModeSuccess := SetBlockModeWithoutClosing(false);
end;

procedure TExtendedTcpClient.Close;
begin
  FIsConnected := false;
  inherited;
end;

function TExtendedTcpClient.SetBlockModeWithoutClosing(Block: Boolean): Boolean;
var
  nonBlock: Integer;
begin
  // TTcpClient.SetBlockMode closes the connection and the socket
  nonBlock := Ord(not Block);
  Result := ErrorCheck(ioctlsocket(Handle, FIONBIO, nonBlock)) <> SOCKET_ERROR;
end;

function TExtendedTcpClient.WaitUntilConnected(Timeout: Integer): Boolean;
var
  writeReady, exceptFlag: Boolean;
begin
  // Select waits until connected or timeout
  Select(nil, @writeReady, @exceptFlag, Timeout);
  Result := writeReady and not exceptFlag;
end;

function TExtendedTcpClient.ConnectWithTimeout(Timeout: Integer): Boolean;
begin
  if Connected or FIsConnected then
    Result := true
  else begin
    if BlockMode = bmNonBlocking then begin
      if Connect then // will return immediately, tipically with false
        Result := true
      else
        Result := WaitUntilConnected(Timeout);
    end
    else begin // blocking mode
      // switch to non-blocking before trying to do the real connection
      FNonBlockingModeRequested := true;
      FNonBlockingModeSuccess := false;
      try
        if Connect then // will return immediately, tipically with false
          Result := true
        else begin
          if not FNonBlockingModeSuccess then
            Result := false
          else
            Result := WaitUntilConnected(Timeout);
        end;
      finally
        if FNonBlockingModeSuccess then begin
          // revert back to blocking
          if not SetBlockModeWithoutClosing(true) then begin
            // undesirable state => abort connection
            Close;
            Result := false;
          end;
        end;
      end;
    end;
  end;
  FIsConnected := Result;
end;

如何使用:

TcpClient := TExtendedTcpClient.Create(nil);
try
  TcpClient.BlockMode := bmBlocking; // can also be bmNonBlocking

  TcpClient.RemoteHost := 'www.google.com';
  TcpClient.RemotePort := '80';

  if TcpClient.ConnectWithTimeout(500) then begin // wait up to 500ms
    ... your code here ...
  end;
finally
  TcpClient.Free;
end;

如前所述,Connected非阻塞套接字不能很好地工作,所以我添加了一个新IsConnected属性来克服这个问题(仅在连接时有效ConnectWithTimeout)。

两者ConnectWithTimeoutIsConnected都适用于阻塞和非阻塞套接字。

于 2013-08-28T20:07:23.070 回答