【问题标题】:How do I correct this LockBits math error?如何更正此 LockBits 数学错误?
【发布时间】:2015-06-02 02:22:28
【问题描述】:

使用 GetPixel/SetPixel,我正在使用以下内容为强调红色和紫色的图像过滤器提供动力:

for (int y = 0; y < bmpMain.Height; y++)
    for (int x = 0; x < bmpMain.Width; x++)
    {
        bmpMain.GetPixel(x, y);
        Color c = bmpMain.GetPixel(x, y);
        int myRed = c.R, myGreen = c.G, myBlue = c.B;
        myGreen -= 128;
        if (myGreen < 0) myGreen = 0;
        bmpMain.SetPixel(x, y, Color.FromArgb(255, myRed, myGreen, myBlue));
    }

使用 LockBits,我已将其替换为以下内容:

for (int counter = 1; counter < rgbValues.Length; counter += 3)
{
    rgbValues[counter] -= 128;
    if (rgbValues[counter] < 0) rgbValues[counter] = 0;  
}

但不是将绿色像素值减去 128,而是将 128添加到绿色值。

如果我这样做:

for (int counter = 1; counter < rgbValues.Length; counter += 3)
{
    rgbValues[counter] += 128;
    if (rgbValues[counter] < 0) rgbValues[counter] = 0;  
}

128 也被添加到绿色值。得到的图像与我减去 128 的图像相同。

那么,我怎样才能让应该简单的数学在 LockBits 中正常工作?

【问题讨论】:

  • 这是因为值是byte 类型。当您减去128 或添加128 时,结果是相同的,因为溢出将返回原始值。 (0 + 128 = 128,0 - 128 = 256 - 128 = 128)图片中的实际值为bytes(不管int你存储为什么类型)。
  • 好的,这解释了问题,谢谢。至于解决方案,我如何减去128? 128 是否需要某种类型的转换或强制转换?我以前从未遇到过这个。实际上,只是尝试强制转换 (byte) 128 和 Convert.ToByte(128),而且我相信您已经知道,两者都没有任何效果。正确的程序是什么?我显然对此完全陌生。谢谢
  • rgbValues[counter] -= (rgbValues[counter] &gt; 128 ? 128 : rgbValues[counter]); 应该如你所愿。

标签: c# image-processing rgb getpixel lockbits


【解决方案1】:

假设rgbValues是一个字节数组,语句

rgbValues[counter] -= 128;

等价于

rgbValues[counter] = (byte)(rgbValues[counter] - 128);

因此,如果 rgbValues[counter] 等于 0,它将被设置为 (byte)(-128)。问题在于,与int 不同,byte 数据类型是 unsigned 并且不能表示负值。正如 EBrown 所说,减法溢出并返回到 128。

修复代码的一种方法是引入int 类型的中间变量,以便您可以安全地容纳负值:

int myGreen = rgbValues[counter];
myGreen -= 128;
if (myGreen < 0) myGreen = 0;  
rgbValues[counter] = (byte)myGreen;

另一种方法是重写代码并首先避免负值:

rgbValues[counter] = rgbValues[counter] > 128
    ? (byte)(rgbValues[counter] - 128)
    : (byte)0;  

【讨论】:

  • 我因工作/家庭问题而离开,对于延迟回复/确认,我深表歉意。非常感谢,这很好用!
  • 我遇到了一个奇怪的问题:调整图像大小后代码不起作用。调整大小和使用 GetPixel/SetPixel 可以正常工作。但是当我使用 LockBits 而不是 Reds 和 Purples 调整大小时,我会得到一系列穿过图像的对角线。关于什么会导致这种情况发生的任何想法,或者如果我无法解决问题,我应该发布一个新问题吗?谢谢
  • 好的,我的调整大小功能正在将图像从 24 位更改为 32 位,这与 GetPixel/SetPixel 无关,但与 Lockbits 无关。我需要回去阅读如何检查 24 位图像和 32 位图像,或者在应用过滤器循环之前转换为 32 位。
  • 你应该发布一个新问题,因为这听起来像是一个不同的问题。
猜你喜欢
  • 2013-08-21
  • 1970-01-01
  • 2014-07-07
  • 1970-01-01
  • 1970-01-01
  • 2021-11-16
  • 1970-01-01
  • 2015-09-24
  • 2021-12-18
相关资源
最近更新 更多