c# - 位图区域已经锁定,多线程环境

标签 c# .net multithreading

我正在使用一个通过多个线程调用主函数的 API。我试图通过该函数访问另一个类中的位图并从中写入,但即使将其设置为使用完全不同的对象实例后,我仍遇到 InvalidOperationException:位图区域已被锁定。

我已经尝试在主函数和调用 Bitmap.LockBits(...) 的地方锁定代码。是的,UnlockBits 在我完成后被调用。

    /* Part of Class B */
    public Surface imageSurface //Surface is a field of pixels, more or less.
    {
        get
        {
            if (_CurrImage != null && _imageSurface == null)
            {

                _imageSurface = Surface.CopyFromBitmap(_CurrImage);
                return Surface.CopyFromBitmap(_imageSurface.CreateAliasedBitmap());
            }
            else
            {
                Surface clearPixel = new Surface(1, 1);
                clearPixel[0, 0] = ColorBgra.FromBgra(255, 255, 255, 0);
                return clearPixel;
            }
        }
    }
    /* the "main" function, Class A */
    public override void Render(ClassBPrototype parameters, ...)
    {
        ClassB token = (ClassB)parameters; // Here we go again~!
        ...
        Surface ourSurface = dstArgs.Surface;
        if (token.imageSurface.Size != null)
        {
            ourSurface = token.imageSurface;
        }

        lock(typeof(ClassA))
        {
            for (int lRectangleIndex = ...)
            {
                Rectangle lRectangle = rois[lRectangleIndex];

                for (int y = ...)
                {
                    surfaceY = (ourSurface.Height / 2) - (y - (int)CenterY);

                    for (int x = ...)
                    {
                        surfaceX = (ourSurface.Width / 2) - (x - (int)CenterX);
                        if (surfaceX >= 0 && surfaceX < ourSurface.Width && surfaceY >= 0 && surfaceY < ourSurface.Height)
                        {
                            dstArgs.Surface[x, y] = ourSurface[surfaceX, surfaceY];
                        }
                        else
                        {
                            dstArgs.Surface[x, y] = ColorBgra.FromBgra(255, 255, 255, 0);
                        }

                    }
                }
            }
        }
    }

最佳答案

问题很可能是共享资源 - 位图 - 不受代码示例中的锁保护。锁需要包装 LockBitsUnlockBits 调用才能有效防止同时访问 Bitmap 实例。

在这种情况下,我建议使用实际的 Bitmap 实例作为锁定对象,因为这是需要保护的,以防止来自不同线程的同时访问。

仅当在静态方法中使用锁时才建议锁定 typeof(ClassA)

请注意,锁是互斥锁,当一个线程获得锁时,其他线程将被挂起。如果大部分时间都花在锁内,那么拥有多个线程将不会或几乎不会带来并行化优势。

在异常情况下,锁定结构的设计将释放锁定,而通常不会调用 UnlockBits 方法。如果可以抛出并捕获异常,我建议从 finally block 调用 UnlockBits 方法。像这样:

private Bitmap _bitmap;
public void foo()
{
    lock (_bitmap)
    {
        BitmapData data;
        try
        {
            data = _bitmap.LockBits(area, ImageLockMOde.ReadOnly, pixelFormat);
            UseBitmapData(data);
        }
        finally
        {
            _bitmap.UnlockBits(data);
        }
    }
}

这个逻辑也可以包装在它自己的实现 IDisposable 接口(interface)的类中,允许使用强大的 using 结构。

关于c# - 位图区域已经锁定,多线程环境,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1552616/

相关文章:

c# - Parallel.ForEach<T> 的某些项目在 ThreadPool 上运行,有些则不

c# - 为 Delphi 应用程序编写 .NET (C#) dll

c# - Asp.Net Core - 错误无法使用主管在已发布的(Ubuntu 14.04)Web 应用程序上执行二进制文件

c# - 将字符串转换为 DataColumn 中的 guid

.net - 使用内联初始化反序列化 bool 数据成员时发生了奇怪的事情

.net - 如何从 PHP 使用 WCF Web 服务?

java - java中如何让线程等待并通知

Java:将 put() 与对象列表同步

c# - 从字符串转换为uniqueidentifier时转换失败

c# - 如何在虚假(C# Faker)规则之间共享状态?