BMP位图的读取与保存

在实现BMP位图的读取与保存之前 我们要了解BMP位图的文件结构 


BMP位图文件结构分为4个部分:

  1 . 位图文件 头数据结构 ,它包含BMP 图像文件的类型、显示内容等信息; 
 2 .位图信息数据结构 ,它包含有BMP 图像的宽、高、压缩方法,以及定义颜色等信息;  
  3. 调色板 ,这个部分是可选的,有些位图需要调色板,有些位图,比如真彩色图(24 位的 BMP )就不需要调色板; 
  4. 位图数据 ,这部分的内容根据BMP 位图使用的位数不同而不同,在 24 位图中直接使用 RGB ,而其他的小于 24 位的使用调色板中颜色索引值。 




① BMP文件头 (14 字节 )

Byte*2 bfType; // 位图文件的类型,必须为 ' B '' M '两个字母 (0-1字节 )  
int bfSize; // 位图文件的大小,以字节为单位 (2-5 字节 )  
int bfReserved1; // 位图文件保留字,必须为 0(6-7 字节 )   
int  bfReserved2; // 位图文件保留字,必须为 0(8-9 字节 )  
int  bfOffBits; // 位图数据的起始位置,以相对于位图 (10-13 字节 )


② 位图信息头(40 字节 )  
nt Size; // 本结构所占用字节数 (14-17 字节 )  
int image_width; // 位图的宽度,以像素为单位 (18-21 字节 )   
int image_heigh; // 位图的高度,以像素为单位 (22-25 字节 )  
short Planes; // 目标设备的级别,必须为 1(26-27 字节 )   
short biBitCount;// 每个像素所需的位数,必须是 1(双色), 4(16 色 ) , 8(256 色 ) 或 24(// 真彩色 ) 之一 (28-29 字节) 
int biCompression; // 位图压缩类型,必须是 0( 不压缩 ),(30-33 字节 ) 1(BI_RLE8 压缩类型 ) 或// 2(BI_RLE4 压缩类型 ) 之一  
int SizeImage; // 位图数据的大小,以字节为单位 (34-37 字节 )   
int biXPelsPerMeter; // 位图水平分辨率,每米像素数 (38-41 字节 )   
int biYPelsPerMeter; // 位图垂直分辨率,每米像素数 (42-45 字节 )   
int biClrUsed;// 位图实际使用的颜色表中的颜色数 (46-49 字节 )   
int biClrImportant;// 位图显示过程中重要的颜色数 (50-53 字节 )
  


③ 颜色表  
颜色表用于说明位图中的颜色,它有若干个表项,每一个表项是一个RGBQUAD 类型的结构,定义一种颜色。
class  RGBQUAD {  
byte  rgbBlue;// 蓝色的亮度 ( 值范围为 0-255)   
byte  rgbGreen; // 绿色的亮度 ( 值范围为 0-255)  
byte  rgbRed; // 红色的亮度 ( 值范围为 0-255)   
byte  rgbReserved;// 保留,必须为 0   
}


由上面定义的RGBQUAD可知  每个表项占据4个字节

颜色表中RGBQUAD 结构数据的个数有 biBitCount 来确定。当biBitCount=1,4,8 时,分别有 2,16,256 个表项 ; 当biBitCount=24 时,没有颜色表项。、
所以当
biBitCount=1时  颜色表占8个字节
biBitCount=4时  颜色表占64个字节
biBitCount=8时  颜色表占1024个字节
biBitCount=24时  颜色表占0个字节

④ 位图数据   
位图数据记录了位图的每一个像素值,记录顺序是在扫描行内是从左到右, 扫描行之间是从下到上。位图的一个像素值所占的字节数:   
当biBitCount=1 时, 8 个像素占 1 个字节 ;   
当biBitCount=4 时, 2 个像素占 1 个字节 ;   
当biBitCount=8 时, 1 个像素占 1 个字节 ;   
当biBitCount=24 时 ,1 个像素占 3 个字节 ;   
Windows规定一个扫描行所占的字节数必须是4 的倍数  不足的以 0 填充 

注:bmp储存数据 都是低位在前 高位在后  而java中是高位在前 低位在后 所以写入读取时要注意转换


// 定义一个内部方法 用来读取bmp整型
public int MyreadInt(BufferedInputStream bis) throws IOException {
		int a = bis.read();
		int b = bis.read();
		int c = bis.read();
		int d = bis.read();
		int e = (d << 24) + (c << 16) + (b << 8) + a;
		return e;
	}  



// ***********************读入24 位BMP文件***************************
public void ReadBMP(BufferedInputStream bis) throws IOException {
		char head[] = new char[2];
		head[0] = (char) bis.read();
		head[1] = (char) bis.read();
		String isbmp = new String(head);
		if ("BM".equals(isbmp)) {
			bis.skip(8);
			// 得到 位图数据的起始位置,
			int loction = MyreadInt(bis);
			bis.skip(4);
			// 得到位图hight width
			int width = MyreadInt(bis);
			int height = MyreadInt(bis);
			drawPanel.setPreferredSize(new Dimension(width, height));
			DrawListener.array = new int[height][width];
			bis.skip(loction - 26);
			int v = 0;
			if (width * 3 % 4 != 0)
				v = 4 - width * 3 % 4;
			for (int i = height - 1; i >= 0; i--) {
				for (int j = 0; j < width; j++) {
					int blue = bis.read();
					int green = bis.read();
					int red = bis.read();
					int rgb = (red << 16) + (green << 8) + blue;
					DrawListener.array[i][j] = rgb;
				}
				bis.skip(v);
			}

		}

		bis.close();
	}




// ***********************读入 16色BMP文件***************************
public void Read16BMP(BufferedInputStream bis) throws IOException {
char head[] = new char[2];
head[0] = (char) bis.read();
head[1] = (char) bis.read();
String isbmp = new String(head);
if ("BM".equals(isbmp)) {
bis.skip(8);
// 得到 位图数据的起始位置,
int loction = MyreadInt(bis);
bis.skip(4);
// 得到位图hight width
int width = MyreadInt(bis);
int height = MyreadInt(bis);
drawPanel.setPreferredSize(new Dimension(width, height));
DrawListener.array = new int[height][width];
bis.skip(loction - 26);

// *********************读位图数据******************************

int v = 0;
			if (width % 2 != 0)
				v = 4 - (width + 1) / 2 % 4;
			else
				v = 4 - width / 2 % 4;
			
			//当width为奇数时
			if (width % 2 != 0) {
				for (int i = height - 1; i >= 0; i--) {
					for (int j = 0; j < width; j++) {
						int a = bis.read();
						if (j == width - 1) {
							int c1 = (a >> 4) & 15;
							DrawListener.array[i][j] = colorArr[c1];
						} else {
							int c1 = (a >> 4) & 15;
							int c2 = a & 15;
							DrawListener.array[i][j] = colorArr[c1];
							j++;
							DrawListener.array[i][j] = colorArr[c2];
						}
					}
					bis.skip(v);
				}
			} else {
				//当width为偶数时
				for (int i = height - 1; i >= 0; i--) {
					for (int j = 0; j < width; j++) {
						int a = bis.read();
						int c1 = (a >> 4) & 15;
						int c2 = a & 15;
						DrawListener.array[i][j] = colorArr[c1];
						j++;
						DrawListener.array[i][j] = colorArr[c2];

					}
					bis.skip(v);
				}
			}

		}

		bis.close();
	}




// 创建16色位图颜色表
public void Creat16color() {
		colorArr[0] = new Color(0, 0, 0).getRGB();
		colorArr[1] = new Color(80, 0, 0).getRGB();
		colorArr[2] = new Color(0, 80, 0).getRGB();
		colorArr[3] = new Color(80, 80, 0).getRGB();
		colorArr[4] = new Color(0, 0, 80).getRGB();
		colorArr[5] = new Color(80, 0, 80).getRGB();
		colorArr[6] = new Color(0, 80, 80).getRGB();
		colorArr[7] = new Color(80, 80, 80).getRGB();
		colorArr[8] = new Color(192, 192, 192).getRGB();
		colorArr[9] = new Color(255, 0, 0).getRGB();
		colorArr[10] = new Color(0, 255, 0).getRGB();
		colorArr[11] = new Color(255, 255, 0).getRGB();
		colorArr[12] = new Color(0, 0, 255).getRGB();
		colorArr[13] = new Color(255, 0, 255).getRGB();
		colorArr[14] = new Color(0, 255, 255).getRGB();
		colorArr[15] = new Color(255, 255, 255).getRGB();

	}





// *******************得到位图文件大小********************
public int GetBMPSize(int type) {
		Dimension dim = drawPanel.getPreferredSize();
		int width = dim.width;
		int height = dim.height;

		int size = 0;
		size += 54;
		if (type == 24) {
			// 24位的时候

			// 计算每行要补充的字节数
			int v = 0;
			if (width * 3 % 4 != 0) {
				v = 4 - width * 3 % 4;
			}
			size += width * height * 3 + v * height;

		} else if (type == 4) {
			int v = 0;
			if ((width / 2) % 4 != 0)
				v = 4 - width / 2 % 4;
			size += width * height / 2 + v * height;
		}
		return size;
	}



// ********************写入int类型**********************
	public void MywriteInt(BufferedOutputStream bos, int data)
			throws IOException {
		int a = data >> 24 & 0xFF;
		byte b = (byte) (data >> 16 & 0xFF);
		byte c = (byte) (data >> 8 & 0xFF);
		byte d = (byte) (data & 0xFF);
		// System.out.println(a+" "+b+" "+c+" "+d);
		// 因为写入时值写入8个字节 所以 只写入 int的最低位 所以无需转型byte
		bos.write(d);
		bos.write(c);
		bos.write(b);
		bos.write(a);
	}

	// ********************写入short类型****************************
	public void MywriteShort(BufferedOutputStream bos, short data)
			throws IOException {
		int a = data >> 8 & 0xFF;
		int b = data & 0xFF;
		bos.write(b);
		bos.write(a);
	}

	// *******************写入BMP文件头 (14 字节 )********************
	public void WriteHead(BufferedOutputStream bos, int type)
			throws IOException {
		byte a = 'B';
		byte b = 'M';
		// 将'B' 'M'写入文件 2个字节
		bos.write(a);
		bos.write(b);
		// 写入位图文件大小 4个字节
		int size = this.GetBMPSize(type);
		this.MywriteInt(bos, size);

		// 位图文件保留字 4个字节 都为0
		bos.write(0);
		bos.write(0);
		bos.write(0);
		bos.write(0);
		// 位图数据的起始位置 4个字节 24位 颜色表为空
		if (type == 24) {
			this.MywriteInt(bos, 54);
		} else if (type == 4) {
			this.MywriteInt(bos, 54 + 64);
		}

	}

	// *******************写入位图信息头(40 字节 ) ********************
	public void WriteMessageHead(BufferedOutputStream bos, int type)
			throws IOException {
		// 本结构所占用字节数 (14-17 字节 )
		this.MywriteInt(bos, 40);
		// 位图的宽度,以像素为单位 (18-21 字节 )
		Dimension dim = drawPanel.getPreferredSize();
		int width = dim.width;
		int height = dim.height;
		// System.out.println(width+"  "+height);
		this.MywriteInt(bos, width);
		this.MywriteInt(bos, height);
		// 目标设备的级别,必须为 1(26-27 字节 )
		this.MywriteShort(bos, (short) 1);
		// 每个像素所需的位数
		this.MywriteShort(bos, (short) type);
		// 位图压缩类型
		this.MywriteInt(bos, 0);
		// 位图数据的大小
		if (type == 24) {
			int size = this.GetBMPSize(type) - 54;
			this.MywriteInt(bos, size);
		} else if (type == 4) {
			int size = this.GetBMPSize(type) - 54 - 64;
			this.MywriteInt(bos, size);
		}
		// 位图水平分辨率 4个字节 位图垂直分辨率 4个字节
		this.MywriteInt(bos, 0);
		this.MywriteInt(bos, 0);
		// 位图实际使用的颜色表中的颜色数
		this.MywriteInt(bos, 16);
		// 位图显示过程中重要的颜色数
		this.MywriteInt(bos, 0);

	}

	// *******************写入位图颜色表 ********************
	public void WriteColor(BufferedOutputStream bos, int type)
			throws IOException {
		if (type == 4) {
			for (int i = 0; i < 16; i++) {
				Color c = new Color(colorArr[i]);
				int red = c.getRed();
				int green = c.getGreen();
				int blue = c.getBlue();
				bos.write(blue);
				bos.write(green);
				bos.write(red);
				bos.write(0);
			}
		}
	}

	// *******************写入位图位图数据********************
	public void WriteData(BufferedOutputStream bos) throws IOException {
		Dimension dim = drawPanel.getPreferredSize();
		int width = dim.width;
		int height = dim.height;
		int v = 0;
		if (width * 3 % 4 != 0)
			v = 4 - width * 3 % 4;
		for (int i = DrawListener.array.length - 1; i >= 0; i--) {
			for (int j = 0; j < DrawListener.array[i].length; j++) {
				int RGB = DrawListener.array[i][j];
				bos.write(RGB & 0xFF);
				bos.write((RGB >> 8) & 0xFF);
				bos.write((RGB >> 16) & 0xFF);

			}
			for (int k = 0; k < v; k++)
				bos.write(0);
		}
	}

你可能感兴趣的:(读取)