找出图像之间的差异

Mou*_*Mou 1 c# image-processing

假设我有两个几乎相同的图像,我想找到并突出它们之间的差异,并产生差异图像.例行工作,但这个例程要求提供我不想要的颜色.这是我的代码.

public class ImageTool
{
    public static unsafe Bitmap GetDifferenceImage(Bitmap image1, Bitmap image2, Color matchColor)
    {
        if (image1 == null | image2 == null)
            return null;

        if (image1.Height != image2.Height || image1.Width != image2.Width)
            return null;

        Bitmap diffImage = image2.Clone() as Bitmap;

        int height = image1.Height;
        int width = image1.Width;

        BitmapData data1 = image1.LockBits(new Rectangle(0, 0, width, height), 
                                           ImageLockMode.ReadOnly, PixelFormat.Format24bppRgb);
        BitmapData data2 = image2.LockBits(new Rectangle(0, 0, width, height), 
                                           ImageLockMode.ReadOnly, PixelFormat.Format24bppRgb);
        BitmapData diffData = diffImage.LockBits(new Rectangle(0, 0, width, height), 
                                               ImageLockMode.WriteOnly, PixelFormat.Format24bppRgb);

        byte* data1Ptr = (byte*)data1.Scan0;
        byte* data2Ptr = (byte*)data2.Scan0;
        byte* diffPtr = (byte*)diffData.Scan0;

        byte[] swapColor = new byte[3];
        swapColor[0] = matchColor.B;
        swapColor[1] = matchColor.G;
        swapColor[2] = matchColor.R;

        int rowPadding = data1.Stride - (image1.Width * 3);

        // iterate over height (rows)
        for (int i = 0; i < height; i++)
        {
            // iterate over width (columns)
            for (int j = 0; j < width; j++)
            {
                int same = 0;

                byte[] tmp = new byte[3];

                // compare pixels and copy new values into temporary array
                for (int x = 0; x < 3; x++)
                {
                    tmp[x] = data2Ptr[0];
                    if (data1Ptr[0] == data2Ptr[0])
                    {
                        same++;
                    }
                    data1Ptr++; // advance image1 ptr
                    data2Ptr++; // advance image2 ptr
                }

                // swap color or add new values
                for (int x = 0; x < 3; x++)
                {
                    diffPtr[0] = (same == 3) ? swapColor[x] : tmp[x];
                    diffPtr++; // advance diff image ptr
                }
            }

            // at the end of each column, skip extra padding
            if (rowPadding > 0)
            {
                data1Ptr += rowPadding;
                data2Ptr += rowPadding;
                diffPtr += rowPadding;
            }
        }

        image1.UnlockBits(data1);
        image2.UnlockBits(data2);
        diffImage.UnlockBits(diffData);

        return diffImage;
    }
}
Run Code Online (Sandbox Code Playgroud)

像这样打电话:

Bitmap diff = ImageTool.GetDifferenceImage(image1, image2, Color.Pink);
diff.MakeTransparent(Color.Pink);
diff.Save("C:\\test-diff.png",ImageFormat.Png);
Run Code Online (Sandbox Code Playgroud)
  1. 有人只是引导我如何改变这个例程,因此当我调用GetDifferenceImage()方法时我们不必传递颜色.

  2. 这种方式图像比较是最好的技术,如果没有那么指导我如何开发一个更快速获得差异图像的例程.

  3. 获得差异图像后,我如何合并差异图像image1.帮助我开发一个更快的合并例程.

den*_*ver 5

如果两个图像相同则差异图像是黑色的,并且对于具有较大差异的像素具有增加的亮度.您可以更改算法,以便不是为交换颜色分配像素,而是为它分配两种颜色之间的差异.

    // iterate over height (rows)
    for (int i = 0; i < height; i++)
    {
        // iterate over width (columns)
        for (int j = 0; j < width; j++)
        {
            // for each channel
            for (int x=0; x<3; x++)
            {
                diffPtr[0] = Abs(data1Ptr[0]-data2Ptr[0]);
                data1Ptr++; // advance image1 ptr
                data2Ptr++; // advance image2 ptr
                diffPtr++; // advance diff image ptr
            }
        }

        // at the end of each column, skip extra padding
        if (rowPadding > 0)
        {
            data1Ptr += rowPadding;
            data2Ptr += rowPadding;
            diffPtr += rowPadding;
        }
    }
Run Code Online (Sandbox Code Playgroud)

你如何显示/合并差异将取决于你将如何处理它.