C# async/await Task<BitmaptImage> WPF 之上的问题

问题描述 投票:0回答:4

小型 WPF 应用程序使用灰度输入参数以异步/等待方式转换图像(BitmapImage)。

我读了很多实现,但没能成功:/

按钮方式:

private async void btnConvertImage_ClickAsync(object sender, RoutedEventArgs e)
{
    try
    {
        var cts = new CancellationTokenSource();
        BitmapImage result = await ImageProcessing.GreyscaleAsync(orginalImage, cts.Token).ConfigureAwait(false);
        imgPhotoConverted.Source = result;
    }
}

灰度任务定义:

public static async Task<BitmapImage> GreyscaleAsync(BitmapImage inputBitmapImage, CancellationToken cancellationToken)
{
    return await Task.Run(() =>
    {
        Bitmap inputBitmap = ToBitmap(inputBitmapImage);
        Bitmap outputImage = new Bitmap(inputBitmap.Width, inputBitmap.Height);
        for (int i = 0; i < inputBitmap.Width; i++)
        {
            for (int x = 0; x < inputBitmap.Height; x++)
            {
                cancellationToken.ThrowIfCancellationRequested();
                Color imageColor = inputBitmap.GetPixel(i, x);
                int grayScale = (int)((imageColor.R * 0.21) + (imageColor.G * 0.72) + (imageColor.B * 0.07));
                Color newColor = Color.FromArgb(imageColor.A, grayScale, grayScale, grayScale);
                outputImage.SetPixel(i, x, newColor);
            }
        }

        return ToBitmapImage(outputImage);
    }, cancellationToken);
}

上线:

imgPhotoConverted.Source = result;

抛出错误:

System.InvalidOperationException: 'The calling thread cannot access this object because a different thread owns it.'
c# bitmap async-await
4个回答
2
投票

您应该在 Stephen Clearys 博客阅读有关 async/await 的更多信息。

遵循那里的建议将引导您找到一个非常明智的解决方案

private async void btnConvertImage_ClickAsync(object sender, RoutedEventArgs e)
{
    var originalImage = ( imgPhotoOriginal.Source as BitmapImage );
    BitmapImage result = await Task.Run( () => originalImage.ToBitmap().ToGrayscale().ToBitmapImage() );
    imgPhotoConverted.Source = result;
}

正在使用这个扩展类

public static class BitmapExtensions
{
    public static Bitmap ToGrayscale( this Bitmap source, CancellationToken cancellationToken = default )
    {
        Bitmap output = new Bitmap( source.Width, source.Height );
        for ( int i = 0; i < source.Width; i++ )
        {
            for ( int x = 0; x < source.Height; x++ )
            {
                cancellationToken.ThrowIfCancellationRequested();
                var imageColor = source.GetPixel( i, x );
                int grayScale = (int)( ( imageColor.R * 0.21 ) + ( imageColor.G * 0.72 ) + ( imageColor.B * 0.07 ) );
                var newColor = System.Drawing.Color.FromArgb( imageColor.A, grayScale, grayScale, grayScale );
                output.SetPixel( i, x, newColor );
            }
        }
        return output;
    }

    public static Bitmap ToBitmap( this BitmapImage source )
    {
        using ( MemoryStream outStream = new MemoryStream() )
        {
            BitmapEncoder enc = new BmpBitmapEncoder();
            enc.Frames.Add( BitmapFrame.Create( source ) );
            enc.Save( outStream );
            System.Drawing.Bitmap bitmap = new System.Drawing.Bitmap( outStream );

            return new Bitmap( bitmap );
        }
    }

    public static BitmapImage ToBitmapImage( this Bitmap source )
    {
        using ( var memory = new MemoryStream() )
        {
            source.Save( memory, ImageFormat.Png );
            memory.Position = 0;

            var bitmapImage = new BitmapImage();
            bitmapImage.BeginInit();
            bitmapImage.StreamSource = memory;
            bitmapImage.CacheOption = BitmapCacheOption.OnLoad;
            bitmapImage.EndInit();
            bitmapImage.Freeze();

            return bitmapImage;
        }
    }
}

1
投票

我认为有一种更简单的方法可以做到这一点:在返回位图之前冻结()位图。

那么哪个线程访问它并不重要(当然实际的 UI 元素仍然需要仅从 WPF 线程访问)

我通过如下修改解决了类似的问题。

Task.Run(() =>
{
...
        var bmp = ToBitMapImage(outputImage);
        bmp.Freeze();
        return bmp;
}...

0
投票

出现异常的原因是您在工作线程内创建

BitmapImage
,然后尝试将其分配给 UI 线程中的
Image.Source
。因此,UI 线程不拥有该对象并且无法与其交互。


-1
投票

我设法解决了它:

  • 我不必使用 BitmapSource,但我必须使用 Freeze() 输出 BitmaImage 并将 result 包装在正确的上下文中:
ThreadPool.QueueUserWorkItem(async delegate
{
    // ThreadPool
    var cts = new CancellationTokenSource();
    BitmapImage result = await ImageProcessing.GreyscaleAsync(orginalImage, cts.Token);
    result.Freeze();

    sc.Post(delegate
    {
        // original context (UI)
        imgPhotoConverted.Source = result;
        cts.Cancel();
    }, null);
}

我希望这对其他人有用。 谢谢!

© www.soinside.com 2019 - 2024. All rights reserved.