网站首页
手机版

Turbo C位图和BMP位图格式分极及应用

更新时间:2006-01-16 00:00:00作者:未知

   近来,图像功能在计算机上的应用十分广泛,一种方便快捷而实用的方法是首先利用图像扫描仪将图像数据自动生成并存入计算机,再利用Windows的PAINTBRUH功能进行加工修改,成为独立的图像文件。但是,仍然有一个问题:PAINTBRUH软件只能在Windows环境下运行。因此要显示一幅图像(.MSP位图、.BMP位图、PCX位图),也要运行庞大的Windows,显然不方便。为此,笔者编写了一个BMP位图文件直接显示在屏幕上的程序。利用Turbo C位图格式,将BMP格式进行转换,实现了这一功能。
一、Turbo C位图格式
Borlond公司的Turbo C是目前在微机上最为流行的C语言版本。它为软件开发者提供了丰富的屏幕操作与图形功能函数。其中getimage( )函数用于将屏幕内的某矩形区域复制到内存缓冲区,putimage( )函数将内存缓冲区中的内容再复制到屏幕上。但由于图像事先还没有出现在屏幕上,所以,不能使用getimage( )函数填写供putimage( )函数显示的内存图像数据。不过可以把图像数据按getimage( )函数产生的格式填于内存缓冲区中,然后调用putimage( )函数,显示内存缓冲区图像,下面分析getimage( )函数产生的格式:
getimage (int letf, int top, int night, int bottm, *buf)
其中,left, top——矩形区域左上角坐标(x,y)。
right, bottom——矩形区域右上角坐标(x,y)。
* buf——指向存储屏幕数据的内存指针。
在内存中,图像数据是按行存放的。头两个字节为图像的宽度,接下去两个字节为图像的高度(均为低字节在前,高字节在后)。后面是真正的图像数据,它以图像的宽度为单位,先是图像的第一行第三位面的内容,然后是第一行第二位面的内容;第一行第一位面的内容;第一行第零位面的内容。第一行完后,接第二行的四个位面,第三行的四个位面等等。当屏幕状态为16色时,则需4个彩色页面。这时,如果图像的宽度为8的整数倍时,则每
行所需的字节数为:
number=(right-left+8) 18*4
如果图像的宽度不为8的倍数据时,则
number=(ceil ((right-left)/8)*4)
其中,ceil (righ-left)8为取大于(right-left)/8的最小整数
则图像数据所需字节总数为:
number * (bottom-top+1)
则所需内存缓冲区字节总数为:
6+number * (bottom -top+1)
这是因为头四个字节存放图像的宽和高,而缓冲区最后2个字节无意义,可以填零。
@@T5S04700.GIF;图 1@@
这样,每行每页面的字节数为:
p=number/4
如图1所示(设宽、高为16×16)
其中,图像的真正数据在每个页面的前m列,即:
当宽度为8的倍数据时,m=P-1,此时每页面的最后一列可为零。而当宽度不为8的倍
数时,m=P。
结合图1 m=2,即数据在每页面的前2列。
二、BMP位图格式
MS Windows的.BMP图像文件可以表示单色或直至24位的彩色图像,.BMP文件是与设备无关的。.BMP文件分为文件首部和文件体两部分。文件首部描述文件和图像的有关参数和彩色表,主体是图像的位图数据。.BMP文件的格式如表1所示。
@@T5S04701.GIF;表1 BMP文件的格式@@
@@T5S04702.GIF;表2 BMP与VGA彩色编码对照@@
.BMP文件的文件体记录图像的位图数据。从图像的左下角开始依次扫描每根扫描线。对于单色,每个象素用1位表示;16色用4位表示一个象素,256色用一个字节,而24位彩色用3个字节。
但.BMP位图的彩色编码与VGA彩色编码不同,必须经过转换才能用C语言在屏幕上显示。表2给出了.BMP位图彩色编码与VGA彩色编码的对照。
三、.BMP位图向Turbo C位图的转换
当VGA视频内存在选择640×480模式时占用4个位面。每个位面上的一位对应一个点。而.BMP位图则采用压缩法的组织形式,两者显然不同。因此,也需要一个变换过程,见图2。
根据这一原理,首先进行彩色编码转换,然后将.BMP位图数转换成C位图格式,写入内存缓冲区的适当位置。
@@T5S04703.GIF;图2 从BMP位图到VGA视频内存的变换过程@@
为了便于参考,笔者编制了一个程序。本程序只要给出.BMP文件的名字,程序会自行在屏幕上显示出来。需说明的是,本程序只能用于小于64K的图像。但也可以将一幅大的.BMP图像分成若干小图像即可。对图像文件格式转换感兴趣的读者,也不难将本程序拓展到自己的应用领域。程序附后。
# include <stdio.h>
# include <graphics.h>
# include <alloc.h>
# include <math.h>
unsigned char convert ();
main()
{
int 1, i, j, k, q, h;
int driver=VGA, mode=VGAHI;
int lenth, high, number, p, m;
unsigned size;
unsigned char ch, a[8], b[4], e[4][8];
FILE *fp; char *s;
fp=fopen (“party. bmp” , “rb”);
fseek (fp, 18, 0); lenth=getw (fp);
/* graphics width */
fseek (fp, 22, 0); high=getw (fp);
#/* graphics high */
#if ((lenth%8)==0) number= (lenth+8)/8*4
#/* bytes of each line */
# else number=4*(ceil ((double)1enth/8));
up=number/4;
mif ((lenth%8)!=0) m=p;
{/* bytes of each line/4 */
else m=p-1;
size=6+number*(high+1)
/* need memory bytes of graphics */
if ((s=char *) malloc (size))==NULL)
{ printf (“malloc error !\n”); exit (0);}
s[0]=lenth & 0x00ff; s[1]=(lenth& 0xff00)>>8;
s[2]=high & 0x00ff; s[3]=(high & 0xff00)>>8;
for (i=4; i<size; i++) s[i]=0;
fseek (fp, 118,0);
#for (l=high-1; 1>=0; 1--)
#for (q=0; q<m; q++) {
# for (k=0; k<4; k++) {
#ch=getc (fp);
uch=convert (ch);
ma[0]=ch&0x1; a[1]=(ch&0x2)>>1;
{a[2]=(ch&0x4)>>2;a[3]=(ch&0x8)>>3;
a[4]=(ch&0x10)>>4;a[5]=(ch&x20)>>5;
a[6]=(ch&0x40)>>6;a[7]=(ch&0x80)>>7;
h=7;
for (j=0; j<2; i++)
for (j=0; j<4; i++) {
e[j][i+2*k]=a[h];
h--;
}
}
#for (i=0; i<4; i++) {
# b[i]=(e[i][0]<<7)+(e[i][1]<<6)+(e[i][2]<<5)+
#(e[i][3]<<4)+(e[i][4]<<3)+(e[i][5]<<2)+
#(e[i][6]<<1)+(e[i][7]);
u s[4+number *1+q]=b[0]; s[4+number *1+q+p]=b[1];
m s[4+number *1+q+2*p]=b[2]; s[4+number *1+q+3*p]=b[3];
{ }
fclose (fp); initgraph (&driver, &mode, “d:\\tc”);
putimage (0, 0, s, 0); getch();
free(s): closegraph();
}
unsigned char convert (unsigned char ch)
/* bmp color convert vga color */
{
unsigned char c[2];
int i;
#c[0]=(ch&0xf0)>>4;
#c[1]=ch&0x0f;
#for (i=0; i<2; i++) {
#if (c[i]==0x01) {c[i]=0x04; continue;}
uif (c[i]==0x03) {c[i]=0x6; continue;}
mif (c[i]==0x04) {c[i]=0x01; continue;}
{if (c[i]==0x06) {c[i]=0x03; continue;}
if (c[i]==0x07) {c[i]=0x8; continue;}
if (c[i]==0x08) {c[i]=0x07; continue;}
if (c[i]==0x09) {c[i]=0xc; continue;}
if (c[i]==0x0b) {c[i]=0xe; continue;}
if (c[i]==0x0c) {c[i]=0x9; continue;}
if (c[i]==0x0e) c[i]=0x0b;
}
ch=(c[0]<<4)+c[1];
return ch;
#}

本文标签:

为您推荐

IT市场初长成

1. 合肥IT业(市场)现状 合肥,位于安徽省中部,可辐射面积大;背靠内陆一些欠发达的地区如大别山区。由于历史、体制等方面的原因,信息化建设起步晚,基础薄,合肥地区生产计算机及其外围设备、通信器材产品的企业很少,基本上完全是一个消费性市场,无论是规模还是

2011-11-10 16:10

定性仿真综述

本文首先介绍了定性仿真的产生背景及理论发展状况,然后说明了定性仿真在各领域的应用情况,最后对定性仿真的发展方向进行了探讨。

2011-11-10 16:06

关于计算机普及教育的几个问题

一、当前计算机普及的形势 一般理论课程采用的方法是∶先理论,后实际;先抽象,后具体;先一般,后个别。我认为对计算机应用课程应当采用的方法是∶从实际到理论;从具体到抽象;从个别到一般;从零碎到系统。事实证明,这样的方法是十分有效的,是符合广大计算机初学

2011-11-10 16:05

新世纪的软件产业与集成电路产业

为推动我国信息产业和集成电路的发展,增强信息产业创新能力和国际竞争能力,带动传统产业改造和产品升级换代,国家实行集成电路税收优惠、软件企业上市优先的政策,国务院已于1999年7月印发了《鼓励软件产业和集成电路产业发展的若干政策》。 为打破几十年来,国内硬

2011-11-10 16:04

定性仿真理论及其应用

本文首先介绍了定性仿真的产生背景及理论发展状况,然后说明了定性仿真在各领域的应用情况,最后对定性仿真的发展方向进行了探讨。

2011-11-10 16:03

浅谈中职学校计算机理论课程教学的改革

计算机 作为现代高科技的产物,其理论知识专业性强。并且,教师不注意教学方法的选择,学生接受起来有很大困难,学生普遍反映计算机理论课程太难、太枯燥。笔者从中职学校的计算机理论课程的培养目标出发,通过对中职学校的计算机理论课程的教学目标与教学现状存在弊端

2011-11-10 16:01

加载中...