C#从整数中绘制位图

时间:2016-01-25 18:02:01

标签: c# bitmap

我必须从整数值中绘制图像。它们存储在List<int>[]中。 该列表有5081个数组,每个数组有2048个值。每个值介于0-1000之间,一个int是一个像素的颜色,因此它是灰度。

我知道如何用setpixel做这件事,但这太慢了。

for (int y = 0; y < channelId.Length; y++) {
            for (int x = 0; x < channelId[y].Count; x++) {
                int myColor = (channelId[y].ElementAt(x) * 255) / 1000;
                if (myColor > 255) {
                    myColor = 255;
                } else if (myColor < 0) {
                    myColor = 0;
                }
                bmp.SetPixel(x, y, Color.FromArgb(myColor, myColor, myColor));
            }
        }

我知道这里有类似的问题如何更快地绘制位图,但他们已经有了位图。我必须从我的价值观中提取图像。

2 个答案:

答案 0 :(得分:2)

如果您不想安全地执行此操作,则可以执行此操作(假设32bppArgb像素格式化):

var rect = new Rectangle(0, 0, bmp.Width, bmp.Height);
var bmpData = bmp.LockBits(rect, ImageLockMode.WriteOnly, bmp.PixelFormat);
IntPtr ptr = bmpData.Scan0;
for (var y = 0; y < channelId.Length; y++)
{
    var scanLineSize = channelId[y].Count*4;
    var rgb = new byte[scanLineSize];
    int idx = 0;
    // Convert the whole scanline
    foreach (var id in channelId[y])
    {
        rgb[idx] = rgb[idx+1] = rgb[idx+2] = (byte)(id*255/1000);
        rgb[idx+3] = 255;
        idx += 4;
    }
    // And copy it in one pass
    System.Runtime.InteropServices.Marshal.Copy(rgb, 0, ptr, scanLineSize);
    ptr += bmpData.Stride;
}
bmp.UnlockBits(bmpData);

如果它不够快,你可以在内存中完成位图的整个转换,并在一次传递中复制整个数组。但这对内存效率来说不是很高,而且对于你正在移动的数据量来说这应该“足够快”。

更新

只是为了好玩,一次通过:

var rect = new Rectangle(0, 0, bmp.Width, bmp.Height);
var bmpData = bmp.LockBits(rect, ImageLockMode.WriteOnly, bmp.PixelFormat);
IntPtr ptr = bmpData.Scan0;
int idx = 0;
var rgb = new byte[channelId[0].Count * 4 * channelId.Length];
foreach (var id in channelId.SelectMany(t => t))
{
    rgb[idx] = rgb[idx+1] = rgb[idx+2] = (byte)(id*255/1000);
    rgb[idx+3] = 255;
    idx += 4;
}
System.Runtime.InteropServices.Marshal.Copy(rgb, 0, ptr, rgb.Length);
bmp.UnlockBits(bmpData);

除了内存效率之外,您还需要确保数组中的所有List<int>包含相同数量的元素,并且位图的Stride与{{1}相同}}。它应该是2048个元素,但你永远不会知道。

答案 1 :(得分:0)

BitmapData bitmapData = bmp.LockBits(
    new Rectangle(0, 0, channelId[0].Count, channelId.Length),
    ImageLockMode.ReadWrite,
    PixelFormat.Format32bppArgb
);
unsafe{
 ColorARGB* startingPosition = (ColorARGB*) bitmapData.Scan0;
 for (int y = 0; y < channelId.Length; y++) {
        for (int x = 0; x < channelId[y].Count; x++) {
            int myColor = (channelId[y].ElementAt(x) * 255) / 1000;
            if (myColor > 255) {
                myColor = 255;
            } else if (myColor < 0) {
                myColor = 0;
            }

            ColorARGB* position = startingPosition + j + i * channelId[0].Count;
            position->A = 255;
            position->R = myColor;
            position->G = myColor;
            position->B = myColor;
        }
    }
    bmp.UnlockBits(bitmapData);
   }