BitmapData类包含如下重要属性:
- Scan0:指定内存区域的首地址;
- Stride:以Bytes形式描述的宽度,是像素数据的一行;
- PixelFormat:数据的真实的像素格式,这是找到正确字节数的关键;
- Width:锁存图像的宽度;
- Height:锁存图像的高度;
图像数据在存储区域内Scan0与Stride的关系如下图所示:
如上图所示,Stride是以字节的方式来描述一行的宽度。为了访问的效率,一行可能不会准确的是像素尺寸的倍数。系统会让像素数据以4字节边界来组成一行,并且会以4字节来补全。例如,每个像素含24bit的图像,若宽度为17个像素,则Stride为52。每一行中的数据占据3*17=51个字节,填充1个字节,这样每一行就有52(13*4)个字节。4BppIndexed(每个字节包含2个像素)的图像,若宽度为17个像素,则其stride为12。其中9个字节,或者8.5个字节为数据,每行用3或4个字节进行填充。
每行数据的布局方式由像素数据的格式决定,RGB图像每像素含24bit,每3个字节为一个像素;RGBA的图像每像素含32bit,每4个字节为一个像素点。而每个字节包含多个像素点的图像类型,例如每4bit为一个像素点或者每1bit为一像素点的图像,必须更加细心的去处理,避免被同一个字节中包含的其他像素点所迷惑。
Stride是一行的宽度,因此我们可以将Y坐标乘以stride后获取某一特定行的起始位置,从而去获取任意一行或者其Y轴坐标。寻找特定行中的某一像素点会更复杂一些,因为我们必须知道当前图像像素格式的布局方式。
下面举例说明在已知像素格式的前提下去访问某一特定像素点的方法:
- Format32BppArgb:给定X/Y坐标点,像素点的第一个元素的地址为Scan0+(y
* stride)+(x*4),指向蓝色字节,接下来三个字节包含绿,红,alpha字节。 - Format24BppRgb:给定X/Y坐标点,像素点的第一个元素的地址为Scan0+(y*Stride)+(x*3),指向蓝色字节,接下来三个字节包含绿,红。
- Format8BppIndexed: 给定X/Y坐标点,字节地址为Scan0+(y*Stride)+x,指向图像的调色板。
- Format4BppIndexed: 给定X/Y坐标点,一个字节包含的的像素点的计算方式如下:Scan0+(y*Stride)+(x/2)。一个字节包含两个像素,高4位bit为左侧像素,低4位bit为右侧像素。每半个字节的bit位可用于从16色调色板中选取颜色。
- Format1BppIndexed: 给定X/Y坐标点,一个字节包含的的像素点的计算方式如下:Scan0+(y*Stride)+(x/8)。每个字节包含8个bit,1个bit位表示1个像素,bit8~bit0分别对应图像由左至右的像素点。这些bit从2色调色板中取值。
分析:
24位或32位Bitmap即每个像素由24位bit或32位bit组成,每个像素点自行保存色彩(RGB或ARGB)如图
8位以下Bitmap是索引位图,即将固定的颜色保存到颜色表中(8位是256色、4位是16色),每个像素点保存的是颜色表中的索引,如图