- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个关于 Task.WaitAll 的问题。起初我尝试使用 async/await 来得到这样的东西:
private async Task ReadImagesAsync(string HTMLtag)
{
await Task.Run(() =>
{
ReadImages(HTMLtag);
});
}
这个函数的内容并不重要,它同步工作,完全独立于外界。
我这样使用它:
private void Execute()
{
string tags = ConfigurationManager.AppSettings["HTMLTags"];
var cursor = Mouse.OverrideCursor;
Mouse.OverrideCursor = System.Windows.Input.Cursors.Wait;
List<Task> tasks = new List<Task>();
foreach (string tag in tags.Split(';'))
{
tasks.Add(ReadImagesAsync(tag));
//tasks.Add(Task.Run(() => ReadImages(tag)));
}
Task.WaitAll(tasks.ToArray());
Mouse.OverrideCursor = cursor;
}
不幸的是,如果我以这种方式使用它(使用 async/await),我会在 Task.WaitAll
上陷入僵局。我的函数完成其工作(因此它们可以正确执行),但 Task.WaitAll
只是永远留在这里,因为显然 ReadImagesAsync 不会返回给调用者。
注释行是实际正确工作的方法。如果我注释 tasks.Add(ReadImagesAsync(tag));
行并使用 tasks.Add(Task.Run(() => ReadImages(tag)));
- 一切正常。
我在这里缺少什么?
ReadImages 方法如下所示:
private void ReadImages (string HTMLtag)
{
string section = HTMLtag.Split(':')[0];
string tag = HTMLtag.Split(':')[1];
List<string> UsedAdresses = new List<string>();
var webClient = new WebClient();
string page = webClient.DownloadString(Link);
var siteParsed = Link.Split('/');
string site = $"{siteParsed[0]} + // + {siteParsed[1]} + {siteParsed[2]}";
int.TryParse(MinHeight, out int minHeight);
int.TryParse(MinWidth, out int minWidth);
int index = 0;
while (index < page.Length)
{
int startSection = page.IndexOf("<" + section, index);
if (startSection < 0)
break;
int endSection = page.IndexOf(">", startSection) + 1;
index = endSection;
string imgSection = page.Substring(startSection, endSection - startSection);
int imgLinkStart = imgSection.IndexOf(tag + "=\"") + tag.Length + 2;
if (imgLinkStart < 0 || imgLinkStart > imgSection.Length)
continue;
int imgLinkEnd = imgSection.IndexOf("\"", imgLinkStart);
if (imgLinkEnd < 0)
continue;
string imgAdress = imgSection.Substring(imgLinkStart, imgLinkEnd - imgLinkStart);
string format = null;
foreach (var imgFormat in ConfigurationManager.AppSettings["ImgFormats"].Split(';'))
{
if (imgAdress.IndexOf(imgFormat) > 0)
{
format = imgFormat;
break;
}
}
// not an image
if (format == null)
continue;
// some internal resource, but we can try to get it anyways
if (!imgAdress.StartsWith("http"))
imgAdress = site + imgAdress;
string imgName = imgAdress.Split('/').Last();
if (!UsedAdresses.Contains(imgAdress))
{
try
{
Bitmap pic = new Bitmap(webClient.OpenRead(imgAdress));
if (pic.Width > minHeight && pic.Height > minWidth)
webClient.DownloadFile(imgAdress, SaveAdress + "\\" + imgName);
}
catch { }
finally
{
UsedAdresses.Add(imgAdress);
}
}
}
}
最佳答案
您正在同步等待任务完成。如果没有一点 ConfigureAwait(false)
魔法,这对 WPF
不起作用。这是一个更好的解决方案:
private async Task Execute()
{
string tags = ConfigurationManager.AppSettings["HTMLTags"];
var cursor = Mouse.OverrideCursor;
Mouse.OverrideCursor = System.Windows.Input.Cursors.Wait;
List<Task> tasks = new List<Task>();
foreach (string tag in tags.Split(';'))
{
tasks.Add(ReadImagesAsync(tag));
//tasks.Add(Task.Run(() => ReadImages(tag)));
}
await Task.WhenAll(tasks.ToArray());
Mouse.OverrideCursor = cursor;
}
如果这是 WPF,那么我确信当某种事件发生时您会调用它。您应该从事件处理程序调用此方法,例如:
private async void OnWindowOpened(object sender, EventArgs args)
{
await Execute();
}
查看问题的编辑版本,我可以看到实际上您可以通过使用 DownloadStringAsync
的异步版本使一切变得非常漂亮:
private async Task ReadImages (string HTMLtag)
{
string section = HTMLtag.Split(':')[0];
string tag = HTMLtag.Split(':')[1];
List<string> UsedAdresses = new List<string>();
var webClient = new WebClient();
string page = await webClient.DownloadStringAsync(Link);
//...
}
现在,tasks.Add(Task.Run(() => ReadImages(tag)));
是怎么回事?
这需要了解SynchronizationContext
。创建任务时,您可以复制调度该任务的线程的状态,以便在完成等待后可以返回该任务。当您调用没有 Task.Run
的方法时,您会说“我想回到 UI 线程”。这是不可能的,因为 UI 线程已经在等待任务,因此它们都在等待自己。当您添加另一个任务时,您是在说:“UI 线程必须调度一个‘外部’任务,该任务将调度另一个‘内部’任务,我稍后会再讨论。”
关于c# - Task.WaitAll 死锁,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51801460/
我听说在多个线程上使用 waitall 时存在限制(要等待的线程数?)。谁能给详细点? 最佳答案 我认为你所指的限制不是线程的数量;它取决于正在等待的句柄 的数量。来自MSDN page对于 Wait
假设我正在编写一个模拟用户在系统上执行某些操作的软件。我正在使用秒表测量完成此类操作所需的时间。 大多数时候这是非常简单的:模拟按钮的点击,一些服务调用与这个按钮相关联。测量完成此服务调用所需的时间。
当我可以让父任务等待它创建的所有任务时,我不确定创建子任务有什么好处。我运行以下代码,它在两种情况下产生了相同的结果。 public static void Main(string[] args) {
我有这样的情况: var retrievalTasks = new Task[2]; retrievalTasks[0] = GetNodesAsync(); retrievalTasks[1] =
我有少量任务,我想使用 Task.WaitAll 并行运行它们。这是我现在拥有的: type System.Threading.Tasks.Task with static member Wait
我有一个关于 Task.WaitAll 的问题。起初我尝试使用 async/await 来得到这样的东西: private async Task ReadImagesAsync(string HTML
我有一个控制台应用程序,我需要在其中从 4 个不同的站点检索一些数据。我将每个 HTTP 请求放在一个任务中,然后等待它们全部完成。 当我只需要从 2 个站点获取数据时,它就可以正常工作。但后来我需要
是否有可能不使用 WaitHandle.WaitAll(waitHandles) 阻止 winForm,而是设置另一个线程,当从 WaitHandle.WaitAll 获取完成信号时将触发该线程? 最
我想同时运行两个任务,其中一个任务中有一个 Task.Delay()。 即一个连续运行,一个每 15 分钟运行一次。 这是我目前所拥有的: public class ContinousAndAggre
为什么这不是每次都完成?相反,它会抛出一个异常,就像没有捕获到异常一样。此外,异常“索引超出数组范围”对我来说没有意义。 int n = 3; string[] nam
我正在尝试使用此异步代码来测试异步关键字: public async Task AsyncMethod() { var link = "http://www.google.com";
我有两个 ContinueWith 用于一个任务。第一个处理任务成功完成的情况,第二个处理任务失败的情况。然后我等到其中一个完成。示例代码如下: var task = Task.Factory.Sta
我正在使用我的可移植类库异步检索一些 rss 文章,该类库使用 Microsoft.Bcl 库(没有 Task.WhenAll)。每篇文章都有一个 rss 评论的 url,我也需要异步检索它。 下面的
我的目标是从 Amazon Web Services 存储桶下载图像。 我有以下代码功能,可以一次下载多张图片: public static void DownloadFilesFromAWS(str
我的代码中有这样一种情况,我开始执行未知数量的任务并想使用 Task.WaitAll()。 像这样: if (condition) { var task1 = Task.Factory.Sta
我创建了一个 Task 列表,如下所示: public void A() { } public void B() { } public void C() { } public void Ex() {
使用 System.Threading.Tasks.Task.WaitAll() 我可以看到我应该使用此方法的可用参数 可见here但是在 visual studio 中编写时,我可以调用完全没有参数
WaitAll 不工作。我在任务完成之前得到完成。 我有一些群组。每个组都包含项目,我希望每个项目在单独的任务中处理。 static void Main(string[] args) { Li
更新以更清楚地解释事情 我有一个运行多个任务的应用程序。有些是最初创建的,有些可以稍后添加。我需要一个编程结构来等待所有任务完成。一旦所有任务完成,应该运行一些其他代码来清理事情并对其他任务生成的数据
就并行性而言,这些是等价的吗? async Task TestMethod1() { Task t1 = GetInt1(); Task t2 = GetInt2(); awa
我是一名优秀的程序员,十分优秀!