gpt4 book ai didi

c# - 如何使 UdpClient.ReceiveAsync() 可取消?

转载 作者:太空狗 更新时间:2023-10-29 17:34:11 28 4
gpt4 key购买 nike

我有一个接口(interface) INetwork用一个方法:

Task<bool> SendAsync(string messageToSend, CancellationToken ct)

接口(interface)的一个实现有这样的代码:

public async Task<bool> SendAsync(string messageToSend, CancellationToken ct)
{
var udpClient = new UdpClient();
var data = Encoding.UTF8.GetBytes (messageToSend);
var sentBytes = await udpClient.SendAsync(data);
return sentBytes == data.Length;
}

不幸的是,SendAsync()UdpClient类不接受 CancellationToken .

所以我开始将其更改为:

public Task<bool> SendAsync(string messageToSend, CancellationToken ct)
{
var udpClient = new UdpClient();
var data = Encoding.UTF8.GetBytes (messageToSend);
var sendTask = udpClient.SendAsync(data);
sendTask.Wait(ct);

if(sendTask.Status == RanToCompletion)
{
return sendTask.Result == data.Length;
}
}

显然这行不通,因为没有 Task被退回。但是,如果我返回任务,签名将不再匹配。 SendAsync()返回 Task<int> , 但我需要一个 Task<bool> .

现在我很困惑。 :-)如何解决?

最佳答案

我知道这有点晚了,但我最近不得不让 UdpClient ReceiveAsync/SendAsync 可取消。

你的第一个代码块是在没有取消的情况下发送的(你的标题说顺便接收......)。

您的第二个代码块绝对不是执行此操作的方法。您正在调用 *Async,然后调用 Task.Wait,它会阻塞直到调用完成。这使得调用有效地同步并且调用 *Async 版本没有意义。最好的解决方案是按如下方式使用 Async:

...
var sendTask = udpClient.SendAsync(data);
var tcs = new TaskCompletionSource<bool>();
using( ct.Register( s => tcs.TrySetResult(true), null) )
{
if( sendTask != await Task.WhenAny( task, tcs.Task) )
// ct.Cancel() called
else
// sendTask completed first, so .Result will not block
}
...

在 UdpClient 上没有内置的取消方法(没有一个函数接受 CancellationToken ),但您可以利用 Task.WhenAny 等待多个任务的能力。 .这将返回第一个完成的任务(这也是使用 Task.Delay() 实现超时的简单方法)。然后我们只需要创建一个将在 CancellationToken 时完成的任务。被取消,我们可以通过创建一个 TaskCompletionSource 来做到这一点并用 CancellationToken 设置它的回调。

一旦取消,我们可以关闭套接字以实际“取消”底层读/写。

最初的想法来自另一个处理文件句柄的 SO 答案,但它也适用于套接字。我通常将其包装在如下扩展方法中:

public static class AsyncExtensions
{
public static async Task<T> WithCancellation<T>( this Task<T> task, CancellationToken cancellationToken )
{
var tcs = new TaskCompletionSource<bool>();
using( cancellationToken.Register( s => ( (TaskCompletionSource<bool>)s ).TrySetResult( true ), tcs ) )
{
if( task != await Task.WhenAny( task, tcs.Task ) )
{
throw new OperationCanceledException( cancellationToken );
}
}

return task.Result;
}
}

然后像这样使用它:

try
{
var data = await client.ReceiveAsync().WithCancellation(cts.Token);
await client.SendAsync(data.Buffer, data.Buffer.Length, toep).WithCancellation(cts.Token);
}
catch(OperationCanceledException)
{
client.Close();
}

关于c# - 如何使 UdpClient.ReceiveAsync() 可取消?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19404199/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com