我写了一个这个函数的版本,它将灰度位图作为蒙板和源位图,并输出一个位图,其中应用了蒙板SetPixel
和GetPixel
,但速度很慢,所以改为我试图写一个使用BitmapData和指针算术,但我得到一个访问冲突,我不知道为什么。在图像蒙板中使用BitmapData访问冲突
当写入结果位图中的像素时,我得到一个AccessViolationException
- 我认为这是由不正确的索引造成的,但我看不到我出错的地方。
public static Bitmap ApplyAlphaMask2(Bitmap source, Bitmap mask)
{
if (source.Size != mask.Size)
{
throw new NotImplementedException("Applying a mask of a different size to the source image is not yet implemented");
}
Bitmap result = new Bitmap(source.Width, source.Height);
unsafe
{
BitmapData source_data = source.LockBits(new Rectangle(0, 0, source.Width, source.Height), ImageLockMode.ReadWrite, PixelFormat.Format32bppArgb);
BitmapData mask_data = mask.LockBits(new Rectangle(0, 0, mask.Width, mask.Height), ImageLockMode.ReadWrite, PixelFormat.Format32bppArgb);
BitmapData result_data = result.LockBits(new Rectangle(0, 0, result.Width, result.Height), ImageLockMode.ReadWrite, PixelFormat.Format32bppArgb);
for(int column = 0; column < source.Height; column++)
{
Int32* source_column_ptr = (Int32*)source_data.Scan0 + (column * Math.Abs(source_data.Stride));
Int32* mask_column_ptr = (Int32*)mask_data.Scan0 + (column * Math.Abs(mask_data.Stride));
Int32* result_column_ptr = (Int32*)result_data.Scan0 + (column * Math.Abs(mask_data.Stride));
for (int row = 0; row < source.Width; row++)
{
Color source_color = Color.FromArgb(source_column_ptr[row]);
Color mask_color = Color.FromArgb(mask_column_ptr[row]);
Int32* result_pixel_ptr = &result_column_ptr[row];
Color result_color = Color.FromArgb((int)(255.0f * mask_color.GetBrightness()), source_color);
*result_pixel_ptr = result_color.ToArgb(); //Access violation!
}
}
source.UnlockBits(source_data);
mask.UnlockBits(mask_data);
result.UnlockBits(result_data);
}
return result;
}
任何帮助,将不胜感激。
编辑:这并不总是发生在同一列,但该行始终显示为0
仅使用Math.Abs是因为示例显示它在此msdn页面上使用:http://msdn.microsoft.com/zh-cn/library/system。 drawing.imaging.bitmapdata.aspx另外,从'mask_data.Stride'改为'result_data.Stride'没有帮助:/ – Ell
步幅是否定的?请检查所有3个步骤。尝试没有吸收。该示例仅使用Abs分配空间,而不是“导航”现有缓冲区。 – usr
@所有文档都包含以下语句:“跨度是单行像素(扫描线)的宽度,四舍五入为四字节边界。如果跨度为正,则位图为自上而下。如果步幅是负值,则位图是自下而上的。“所以这似乎是错误 - 位图是自下而上的。 – usr