异步等待线程完成

时间:2013-01-12 23:52:40

标签: c# multithreading

我有一个从电脑截取屏幕截图的功能,但遗憾的是它阻止了主UI,所以我决定对它进行异步[线程调用];但是,我发现在返回之前等待Thread的结果很麻烦位图。

这是我的代码:

/// <summary>
/// Asynchronously uses the snapshot method to get a shot from the screen.
/// </summary>
/// <returns> A snapshot from the screen.</returns>
private Bitmap SnapshotAsync()
{
    Bitmap image = null;
    new Thread(() => image = Snapshot()).Start();

    while (image == null)
    {
        new Thread(() => Thread.Sleep(500)).Start(); //Here i create new thread to wait but i don't think this is a good way at all.
    }
    return image;
}

/// <summary>
/// Takes a screen shots from the computer.
/// </summary>
/// <returns> A snapshot from the screen.</returns>
private Bitmap Snapshot()
{
    var sx = Screen.PrimaryScreen.Bounds.Width;
    var sy = Screen.PrimaryScreen.Bounds.Height;
    var shot = new Bitmap(sx, sy, PixelFormat.Format32bppArgb);
    var gfx = Graphics.FromImage(shot);
    gfx.CopyFromScreen(0, 0, 0, 0, new Size(sx, sy));
    return shot;
}

虽然上面的方法是我想要的异步工作,但我相信它可以改进。特别是我执行数百个线程等待结果的方式,我确信这种方式并不好。

所以我真的需要任何一个看一下代码并告诉我如何改进代码的人。

[注意我使用的是.NET 3.5]

提前谢谢。

在Eve和SiLo的帮助下解决的问题是最好的2个答案

  
      
  • 1:
  •   
>     private void TakeScreenshot_Click(object sender, EventArgs e)
>     {
>       TakeScreenshotAsync(OnScreenshotTaken);
>     }
>     
>     private static void OnScreenshotTaken(Bitmap screenshot)
>     {
>       using (screenshot)
>         screenshot.Save("screenshot.png", ImageFormat.Png);
>     }
>     
>     private static void TakeScreenshotAsync(Action<Bitmap> callback)
>     {
>       var screenRect = Screen.PrimaryScreen.Bounds;
>       TakeScreenshotAsync(screenRect, callback);
>     }
>     
>     private static void TakeScreenshotAsync(Rectangle bounds, Action<Bitmap> callback)
>     {
>       var screenshot = new Bitmap(bounds.Width, bounds.Height,
>                                   PixelFormat.Format32bppArgb);
>     
>       ThreadPool.QueueUserWorkItem((state) =>
>       {
>         using (var g = Graphics.FromImage(screenshot))
>           g.CopyFromScreen(bounds.X, bounds.Y, 0, 0, bounds.Size);
>     
>         if (callback != null)
>           callback(screenshot);
>       });
>     }
  
      
  • 2:
  •   
>     void SnapshotAsync(Action<Bitmap> callback)
>     {
>         new Thread(Snapshot) {IsBackground = true}.Start(callback);
>     }
  

>     void Snapshot(object callback)
>     {
>         var action = callback as Action<Bitmap>;
>         var sx = Screen.PrimaryScreen.Bounds.Width;
>         var sy = Screen.PrimaryScreen.Bounds.Height;
>         var shot = new Bitmap(sx, sy, PixelFormat.Format32bppArgb);
>         var gfx = Graphics.FromImage(shot);
>         gfx.CopyFromScreen(0, 0, 0, 0, new Size(sx, sy));
>         action(shot);
>     }
  

用法,例如,通过按钮点击:

void button1_Click(object sender, EventArgs e)
{
    SnapshotAsync(bitmap => MessageBox.Show("Copy successful!"));
}

4 个答案:

答案 0 :(得分:2)

async / await个关键字完全符合您的要求,非常优雅。

以下是将方法转换为正确模式的方法:

private static async Task<Bitmap> TakeScreenshotAsync()
{
  var screenRect = Screen.PrimaryScreen.Bounds;
  return await TakeScreenshotAsync(screenRect);
}

private static async Task<Bitmap> TakeScreenshotAsync(Rectangle bounds)
{
  var screenShot = new Bitmap(bounds.Width, bounds.Height, 
                              PixelFormat.Format32bppArgb);

  // This executes on a ThreadPool thread asynchronously!
  await Task.Run(() =>
  {
    using (var g = Graphics.FromImage(screenShot))
      g.CopyFromScreen(bounds.X, bounds.Y, 0, 0, bounds.Size);

  });

  return screenShot;
}

然后你会做这样的事情:

private async void TakeScreenshot_Click(object sender, EventArgs e)
{
  var button = sender as Button;
  if(button == null) return;

  button.Enabled = false;
  button.Text = "Screenshoting...";

  var bitmap = await TakeScreenshotAsync();
  bitmap.Save("screenshot.png", ImageFormat.Png);

  button.Text = "Take Screenshot";
  button.Enabled = true;
}

答案 1 :(得分:2)

您可以使用基于事件的异步模式:

void SnapshotAsync(Action<Bitmap> callback)
{
    new Thread(Snapshot) {IsBackground = true}.Start(callback);
}

void Snapshot(object callback)
{
    var action = callback as Action<Bitmap>;
    var sx = Screen.PrimaryScreen.Bounds.Width;
    var sy = Screen.PrimaryScreen.Bounds.Height;
    var shot = new Bitmap(sx, sy, PixelFormat.Format32bppArgb);
    var gfx = Graphics.FromImage(shot);
    gfx.CopyFromScreen(0, 0, 0, 0, new Size(sx, sy));
    action(shot);
}

用法,例如,通过按钮点击:

void button1_Click(object sender, EventArgs e)
{
    SnapshotAsync(bitmap => MessageBox.Show("Copy successful!"));
}

正如作者所要求的,它不会阻止原始线程。如果你必须通过回调操作UI,请注意,记住使用Invoke及其等价物。

编辑:阅读SiLo对一些可以应用于上述代码的良好做法和优化的评论。

答案 2 :(得分:2)

我刚刚看到您关于使用3.5而不是4.5的编辑。这太糟糕了,但绝对有可能。我已经创建了第二个答案,因此使用async / await的人可以使用第一个答案作为示例。

现在为你的解决方案,它真的没有太大的不同:

private void TakeScreenshot_Click(object sender, EventArgs e)
{
  TakeScreenshotAsync(OnScreenshotTaken);
}

private static void OnScreenshotTaken(Bitmap screenshot)
{
  using (screenshot)
    screenshot.Save("screenshot.png", ImageFormat.Png);
}

private static void TakeScreenshotAsync(Action<Bitmap> callback)
{
  var screenRect = Screen.PrimaryScreen.Bounds;
  TakeScreenshotAsync(screenRect, callback);
}

private static void TakeScreenshotAsync(Rectangle bounds, Action<Bitmap> callback)
{
  var screenshot = new Bitmap(bounds.Width, bounds.Height,
                              PixelFormat.Format32bppArgb);

  ThreadPool.QueueUserWorkItem((state) =>
  {
    using (var g = Graphics.FromImage(screenshot))
      g.CopyFromScreen(bounds.X, bounds.Y, 0, 0, bounds.Size);

    if (callback != null)
      callback(screenshot);
  });
}

答案 3 :(得分:0)

很抱歉,但从逻辑上讲,这不是太聪明了。

  • 你想拍一张截图。
  • 你不希望UI线程被阻止,所以你去异步。

Gratulations。到目前为止,这是有道理的。

现在出现了你不想告诉任何你试过的人的部分:

  • 现在您要在UI线程中等待异步操作完成。

回到我们的开始 - 你阻止了UI线程。什么都没有实现。你基本上在逻辑上最终与你开始的地方完全相同。

好的,解决方案:

  • 首先,摆脱线程,使用一个Task。效率更高。
  • 其次,意识到在UI线程中等待是没有意义的。取消激活UI元素,然后在处理结束时将其重新打开。

将此处理为状态机问题(UI处于“工作”状态或“等待命令”状态),因此您不会阻止。这是处理这种情况的唯一方法 - 因为如果你等待执行完成,那么整个异步orpeation就没用了。

你无法启动一个方法,然后等待处理完成阻塞线程 - 如果你尝试过,那么整个异步操作就是一个无用的命题。