- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
给定以下代码,为什么 ask.WhenAny
在提供 1 秒的 Task.Delay
时从不返回?从技术上讲,我不确定它是否会在很长一段时间后返回,但它不会在 15 秒左右后返回,之后我手动终止了该进程。根据文档,我不需要手动启动 delayTask
,事实上,如果我尝试手动启动,我会收到一个异常。
当用户在 WPF 应用程序中选择上下文菜单项时,将从 UI 线程调用代码,但如果我为上下文菜单项指定了单击方法,则它可以正常工作在新线程中运行此代码。
public void ContextMenuItem_Click(object sender, RoutedEventArgs e)
{
...
SomeMethod();
...
}
public void SomeMethod()
{
...
SomeOtherMethod();
....
}
public void SomeOtherMethod()
{
...
TcpClient client = Connect().Result;
...
}
//In case you're wondering about the override below, these methods are in
//different classes i've just simplified things here a bit so I'm not posting
//pages worth of code.
public override async Task<TcpClient> Connect()
{
...
Task connectTask = tcpClient.ConnectAsync(URI.Host, URI.Port);
Task delayTask = Task.Delay(1000);
if (await Task.WhenAny(connectTask, delayTask) == connectTask)
{
Console.Write("Connected\n");
...
return tcpClient;
}
Console.Write("Timed out\n");
...
return null;
}
如果我将 ContextMenuItem_Click 更改为以下内容,它就可以正常工作
public void ContextMenuItem_Click(object sender, RoutedEventArgs e)
{
...
new Thread(() => SomeMethod()).Start();
...
}
最佳答案
我预测在您的调用堆栈中,您正在调用 Task.Wait
或 Task<T>.Result
.这将 cause a deadlock我在我的博客上做了完整的解释。
简而言之,发生的是 await
将(默认情况下)捕获当前“上下文”并使用它来恢复其 async
方法。在此示例中,“上下文”是 WPF UI 上下文。
因此,当您的代码执行其 await
时在 WhenAll
返回的任务上,它捕获 WPF UI 上下文。稍后,当该任务完成时,它将尝试在 UI 线程上恢复。但是,如果 UI 线程被阻塞(即在调用 Wait
或 Result
时),则 async
方法无法继续运行并且永远不会完成它返回的任务。
正确的解决方案是使用 await
而不是 Wait
或 Result
.这意味着您的调用代码需要是 async
,它将通过您的代码库传播。最终,您需要决定如何使您的 UI 异步,这本身就是一门艺术。至少开始时,您需要一个 async void
事件处理程序或某种异步 MVVM 命令(我探索 async MVVM commands in an MSDN article )。从那里你需要设计一个合适的异步用户界面;即,当异步操作正在进行时,您的 UI 的外观以及它允许的操作。
关于C#/.NET 4.5 - 为什么在 WPF 应用程序的 UI 线程中提供 Task.Delay 时 "await Task.WhenAny"永远不会返回?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23984655/
我有一个控制台应用程序,我在其中使用 TPL,当我在任务中点击 whenany 时它会退出。我是线程的新手,有人可以指导我正确的方向吗(我单独执行了 RunJob,它没有异常(exception))。
当将多个已完成的任务传递给 Task.WhenAny 时,Task.WhenAny 会优先选择将返回哪个已完成的任务? 最佳答案 当您想知道确切的行为时,通常可以查看 reference source
考虑一个具有某种生命周期的类。在此生命周期中,一个事件可能会发生任意多次,并且该事件是通过完成任务(在事件发生后更新)来发出信号的。该对象也可能被关闭,结束其生命周期。关闭也通过完成任务发出信号。它也
我有这段代码: var client = new TcpClient(); HttpRequestInfo.AddTimestamp("Connecting"); await Task.WhenAny
我在 ReactiveLists 中保存了一些检查列表,它们具有 ChangeTrackingEnabled = true。我只想在每个列表中至少选中一个项目时才启用我的 OkCommand。 此外,
在下面的代码中: if (await Task.WhenAny(task, Task.Delay(100)) == task) { success = true
我想同时执行多个异步任务。每个任务将运行一个 HTTP 请求,该请求可以成功完成或引发异常。我需要等待直到第一个任务成功完成,或者直到所有任务都失败。 如何实现 Task.WhenAny 的重载接受谓
当我使用 Task.WhenAll() 函数并在任务中抛出异常时,会抛出一个新的 AggregateException,我可以捕获它以查看任务中发生的所有异常。但是,当我使用 Task.WhenAny
在我的应用程序中,我正在创建一些并发的网络请求,当其中任何一个完成时我都很满意,所以我使用了方法 Task.WhenAny : var urls = new string[] { "https
假设我有三个任务,a、b 和 c。这三个都保证在 1 到 5 秒之间的随机时间抛出异常。然后我写了下面的代码: await Task.WhenAny(a, b, c); 这最终会从最先出错的任务中抛出
我有以下代码: List> tasks = tasksQuery.ToList(); while (tasks.Any()) { Task completedTask = await Task
我需要为移动应用程序中的任务调用添加超时功能。我尝试使用 Task.WhenAny 来完成此操作,如下所示。这将返回首先完成的任务。我的问题是,最初我是从这个任务中获取返回值的,如果任务没有超时,我仍
我有以下代码片段来处理 Azure 通知中心推送通知: var alert = "{\"aps\":{\"alert\":\"" + message + "\"}}"; var task = Azur
我有一个带有两个 WhenAnyValues 的对象: this.WhenAnyValue(x => x.SomeInt) .Select(x => "Bar" + x) .ToPro
我的解决方案中有两个项目:WPF 项目和类库。 在我的类库中: 我有一个符号列表: class Symbol { Identifier Identifier {get;set;}
我有一个带有两个 WhenAnyValues 的对象: this.WhenAnyValue(x => x.SomeInt) .Select(x => "Bar" + x) .ToPro
我如何并行运行 2 个任务并在一个特定任务结束时得到通知(在我的例子中是列表中的第一个任务) 目前我正在使用这段代码: await Task.WhenAny(A.Play(cancel), B.Pla
我有一个创建多个基于 I/O 的任务的解决方案,我正在使用 Task.WhenAny() 来管理这些任务。但往往很多任务会因为网络问题或请求节流等原因而失败。 在使用 Task.WhenAny() 方
考虑在 ReactiveUI 中使用继承。我有带有 DoSomethingCommand 的 ViewModel 基类。此命令的“CanExecute”取决于属性 Prop1 public class
Reactive UI 中有几种扩展方法可用于获取属性更改的 observable。 我想我明白 WhenAny和 WhenAnyValue . WhenAny用于一系列属性更改通知,您希望其中的对象
我是一名优秀的程序员,十分优秀!