c# - 根据亮度设置黑白像素

标签 c# bitmap lockbits

我正试图从这些或类似的图片中找到针:

我的解决方案是取图片的平均亮度并根据它设置黑白像素。结果是这样的:

我不需要在那里看到数字和工作人员.. 只是针,但当它在那里时这不是什么大问题。但是我使用 set,get pixel 函数,它真的很慢谁能帮我处理我的代码?..thnx

BitmapData imageData = imgPristroj.LockBits(new Rectangle(0, 0, imgPristroj.Width, imgPristroj.Height), ImageLockMode.ReadOnly, imgPristroj.PixelFormat);
        float brightness = 0;
        float average;
        unsafe
        {
            try
            {
                UnmanagedImage Unimg = new UnmanagedImage(imageData);
                int pixelSize = (Unimg.PixelFormat == PixelFormat.Format24bppRgb) ? 3 : 4;
                byte* p = (byte*)Unimg.ImageData.ToPointer();

                for (int y = 0; y < Unimg.Height; y++)
                {
                    for (int x = 0; x < Unimg.Width; x++, p += pixelSize)
                    {
                        brightness += Unimg.GetPixel(x, y).GetBrightness();
                    }
                }
                average = brightness / (Unimg.Width * Unimg.Height);
            }
            finally
            {
                imgPristroj.UnlockBits(imageData); //Unlock
            }
        }


        img19 = (Bitmap)imgPristroj.Clone();

        for (int y = 0; y < img19.Height; y++)
        {
            for (int x = 0; x < img19.Width; x++)
            {

                if (img19.GetPixel(x, y).GetBrightness() > average)
                {
                    img19.SetPixel(x, y, Color.White);
                }
                else
                {
                    img19.SetPixel(x, y, Color.Black);

                }

            }
        }

Edit2:这是我的全部代码..

        float brightness = 0;
        float average = 0;

        PixelUtil pixelUtil2 = new PixelUtil((Bitmap)imgPristroj.Clone());
        pixelUtil2.LockBits();

        for (int y = 0; y < imgPristroj.Height; y++)
        {
            // for each pixel
            for (int x = 0; x < imgPristroj.Width; x++)
            {
                brightness += pixelUtil2.GetPixel(x, y).GetBrightness();
            }
        }
        average = brightness / (imgPristroj.Width * imgPristroj.Height);

        pixelUtil2.UnlockBits();

        img19 = (Bitmap)imgPristroj.Clone();
        Crop cfilter1 = new Crop(new Rectangle(0, (int)(pix * 1.6), img19.Width, (int)(pix * 3)));
        img19 = cfilter1.Apply(img19);

        PixelUtil pixelUtil = new PixelUtil(img19);
        pixelUtil.LockBits();


        for (int y = 0; y < img19.Height; y++)
        {
            for (int x = 0; x < img19.Width; x++)
            {

                if (pixelUtil.GetPixel(x, y).GetBrightness() >= average)
                {
                    pixelUtil.SetPixel(x, y, Color.White);
                }
                else
                {
                    pixelUtil.SetPixel(x, y, Color.Black);
                }

            }
        }

        pixelUtil.UnlockBits();

        string filepath = Environment.CurrentDirectory;
        string fileName = System.IO.Path.Combine(filepath, @"img" + ".bmp");
        img19.Save(fileName);

我得到这个带有彩色像素的位图,你能告诉我为什么吗?

当我使用红色时.. 不是黑色... (pixelUtil.SetPixel(x, y, Color.Red);) 我得到了这张有趣的照片。 (不同尺寸都可以..)

最佳答案

您可以尝试使用编码(marshal)处理:它非常快。 (你只需要复制粘贴这个)

public class PixelUtil
{
    Bitmap source = null;
    IntPtr Iptr = IntPtr.Zero;
    BitmapData bitmapData = null;

    public byte[] Pixels { get; set; }
    public int Depth { get; private set; }
    public int Width { get; private set; }
    public int Height { get; private set; }

    /// <summary>
    /// Pixel marshaling class, use this to get and set pixels rapidly.
    /// </summary>
    /// <param name="source">The Bitmap to work with</param>
    public PixelUtil(Bitmap source)
    {
        this.source = source;
    }

    /// <summary>
    /// Lock bitmap data
    /// </summary>
    public void LockBits()
    {
        try
        {
            // Get width and height of bitmap
            Width = source.Width;
            Height = source.Height;

            // get total locked pixels count
            int PixelCount = Width * Height;

            // Create rectangle to lock
            var rect = new Rectangle(0, 0, Width, Height);

            // get source bitmap pixel format size
            Depth = System.Drawing.Image.GetPixelFormatSize(source.PixelFormat);

            // Check if bpp (Bits Per Pixel) is 8, 24, or 32
            if (Depth != 8 && Depth != 24 && Depth != 32)
            {
                throw new ArgumentException("Only 8, 24 and 32 bpp images are supported.");
            }

            // Lock bitmap and return bitmap data
            bitmapData = source.LockBits(rect, ImageLockMode.ReadWrite,
                                         source.PixelFormat);

            // create byte array to copy pixel values
            int step = Depth / 8;
            Pixels = new byte[PixelCount * step];
            Iptr = bitmapData.Scan0;

            // Copy data from pointer to array
            Marshal.Copy(Iptr, Pixels, 0, Pixels.Length);
        }
        catch (Exception)
        {
            throw;
        }
    }

    /// <summary>
    /// Unlock bitmap data
    /// </summary>
    public void UnlockBits()
    {
        try
        {
            // Copy data from byte array to pointer
            Marshal.Copy(Pixels, 0, Iptr, Pixels.Length);

            // Unlock bitmap data
            source.UnlockBits(bitmapData);
        }
        catch (Exception)
        {
            throw;
        }
    }

    /// <summary>
    /// Get the color of the specified pixel
    /// </summary>
    /// <param name="x"></param>
    /// <param name="y"></param>
    /// <returns></returns>
    public Color GetPixel(int x, int y)
    {
        Color clr = Color.Empty;

        // Get color components count
        int cCount = Depth / 8;

        // Get start index of the specified pixel
        int i = ((y * Width) + x) * cCount;

        if (i > Pixels.Length - cCount)
            throw new IndexOutOfRangeException();

        if (Depth == 32) //For 32 bpp get Red, Green, Blue and Alpha
        {
            byte b = Pixels[i];
            byte g = Pixels[i + 1];
            byte r = Pixels[i + 2];
            byte a = Pixels[i + 3]; // a
            clr = Color.FromArgb(a, r, g, b);
        }
        if (Depth == 24) //For 24 bpp get Red, Green and Blue
        {
            byte b = Pixels[i];
            byte g = Pixels[i + 1];
            byte r = Pixels[i + 2];
            clr = Color.FromArgb(r, g, b);
        }
        if (Depth == 8) //For 8 bpp get color value (Red, Green and Blue values are the same)
        {
            byte c = Pixels[i];
            clr = Color.FromArgb(c, c, c);
        }
        return clr;
    }

    /// <summary>
    /// Set the color of the specified pixel
    /// </summary>
    /// <param name="x"></param>
    /// <param name="y"></param>
    /// <param name="color"></param>
    public void SetPixel(int x, int y, Color color)
    {
        //Get color components count
        int cCount = Depth / 8;

        //Get start index of the specified pixel
        int i = ((y * Width) + x) * cCount;

        if (Depth == 32) //For 32 bpp set Red, Green, Blue and Alpha
        {
            Pixels[i] = color.B;
            Pixels[i + 1] = color.G;
            Pixels[i + 2] = color.R;
            Pixels[i + 3] = color.A;
        }
        if (Depth == 24) //For 24 bpp set Red, Green and Blue
        {
            Pixels[i] = color.B;
            Pixels[i + 1] = color.G;
            Pixels[i + 2] = color.R;
        }
        if (Depth == 8) //For 8 bpp set color value (Red, Green and Blue values are the same)
        {
            Pixels[i] = color.B;
        }
    }
}

你可以这样调用:

public void Example(Bitmap image)
{
  PixelUtil pixelUtil = new PixelUtil(image);
  pixelUtil.LockBits();

  Color firstPixel = pixelUtil.GetPixel(0, 0);

  pixelUtil.SetPixel(0, 0, Color.White);

  //Don't forget to unlock!
  pixelUtil.UnlockBits();
}

编辑:

不确定你是如何保存图像的,但它对我来说很好用:

    public Form1()
    {
        InitializeComponent();

        Bitmap image1 = new Bitmap(@"C:\Users\Nicke Manarin\Desktop\YEeJO.png");

        BlackWhite(image1);
    }

    public void BlackWhite(Bitmap image)
    {
        PixelUtil pixelUtil = new PixelUtil(image);
        pixelUtil.LockBits();

        for (int y = 0; y < image.Height; y++)
        {
            for (int x = 0; x < image.Width; x++)
            {
                if (pixelUtil.GetPixel(x, y).GetBrightness() > 0.5)
                {
                    pixelUtil.SetPixel(x, y, Color.White);
                }
                else
                {
                    pixelUtil.SetPixel(x, y, Color.Black);
                }
            }
        }

        pixelUtil.UnlockBits();

        pictureBox1.Image = image;
        image.Save(@"C:\Users\Nicke Manarin\Desktop\YEeJO2.png");
    }

enter image description here

关于c# - 根据亮度设置黑白像素,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22257878/

相关文章:

c++ - Qt 中的锁位。如何实现?

c# - GCM ASP.NET Android 教程

c# - PowerShell 中的“无法转换 "System.__ComObject"值 ...”错误

java.lang.outofmemoryerror android.graphics.BitmapFactory.nativeDecodeAsset(本地方法)

java - 如何在下载图像之前确定图像大小和尺寸

c# - 位图数据和 Marshal.Copy? Windows Phone 有什么替代品?

c# - XML 序列化异常

c# - 为什么我的 Interop 代码会抛出 "Stack cookie instrumentation code detected a stack-based buffer overrun"异常?

c++ - Fread 跳过字符读入对象

c# - LockBits 似乎对我的需求来说太慢了 - 替代方案?