C语言代码创建、解析BMP格式图片

简介: BMP格式的图片是众多图片格式中的一种,也称为位图数据,BMP结构也比较简单,不需要依赖任何外部库,直接手撸几十行代码即可完成解码编码,非常方便。

一、BMP图片格式介绍

BMP格式的图片是众多图片格式中的一种,也称为位图数据。通常BMP图片是没有压缩的,内部存放的是原始RGB数据,所以BMP文件本身占用的空间比较大。目前在CPU强大的设备上,最常用的格式都是JPG格式,JPG属于压缩格式,占用空间比较小,CPU强大就不在乎压缩和解压消耗的时间。在嵌入式设备上,CPU性能一般较弱,如果要显示图片,使用JPG格式就比较慢,不合适,解码消耗时间太长,造成卡顿,这时候就可以采用BMP格式的图片,不需要解码,直接按照BMP格式的结构读取RGB图形数据即可。而且BMP结构也比较简单,不需要依赖任何外部库,直接手撸几十行代码即可完成解码编码,非常方便。

典型的BMP图像文件由四部分组成:
`
1:文件头
2:图像参数
3:调色板
4:位图数据
`
现在比较常用的是24位真彩色图片,24位真彩色图片就只有3个部分,分别是: 文件头、图像参数、位图数据。这篇文章就介绍24位真彩色(RGB888)的BMP图片如何解码编码。

下面是BMP图片的存储结构:

  1. 文件头: 它包含BMP图像文件的类型、内容尺寸和起始偏移量等信息;
字节顺序 数据结构 描述
1,2 short 高8位为字母’B’,低8位为字母’M’
3,4,5,6 int 文件大小
7,8 short 保留字1
9,10 short 保留字2
11,12,13,14 int 数据部分偏移量
  1. 图像参数,它包含图像的宽、高、压缩方法,以及颜色定义等信息;
字节顺序 数据结构 描述
15,16,17,18 int 当前结构体的大小,通常是40或56
19,20,21,22 int 图像宽度(像素) 0x12~0x15是宽
23,24,25,26 int 图像高度(像素) 0x16~0x19是宽
27,28 short 这个字的值永远是1 说的是两个字节总和是1,
29,30(0x18,0x19) short 每像素占用的位数,即bpp 每个像素所需的位数,必须是1(双色)、4(16色)、8(256色)、24(真彩色)之一
31,32,33,34 int 压缩方式 0x1e~0x21,值是0表示不压缩
35,36,37,38 int 水平分辨率,pixels-per-meter
39,40,41,42 int 垂直分辨率,pixels-per-meter
43,44,45,46 int 垂直分辨率,pixels-per-meter
47,48,49,50 int 引用色彩数
51,52,53,54 int 关键色彩数
  1. 位图数据

位图数据存放的位置由文件头里的第5个参数决定(位图数据偏移量),正常情况下,位图数据就紧接着存放在图像参数的后面。

读取或者写入位图数据的注意事项:

(1) 每行的字节数必须是4的倍数,如果不是,则需要用0补齐。

(2) BMP位图数据的存放是从下到上,从左到右的。先读最后一行,读完后在读倒数第二行。

二、示例代码

2.1 读取BMP图片的参数信息

#include <stdio.h>
#include <string.h>
#include <stdlib.h>

#pragma pack(1) //强制1个字节对齐
//BMP的文件头
struct _BMP_HEAD
{
    char type[2]; //图片的类型 "BM"
    unsigned int size; //文件大小
    unsigned short  r1; //保留1
    unsigned short  r2; //保留2
    unsigned int seek; //数据偏移字节(真实像素点数据)
};

//BMP的参数信息
struct _BMP_INFO
{
    unsigned int size; //当前结构体大小
    unsigned int w; //宽度
    unsigned int h; //高度
    unsigned short flag; //固定为1
    unsigned short bit; //像素点的位数
    unsigned int r1; //压缩方式  0
    unsigned int r2; //水平分辨率
    unsigned int r3; //垂直分辨率
    unsigned int r4; //垂直分辨率
    unsigned int r5; //引用色彩
    unsigned int r6; //关键色彩
};

int main(int argc,char **argv)
{
    if(argc!=2)
    {
        printf("传入的参数格式: ./a.out <文件名称>\n");
        return 0;
    }
    
    /*1. 打开BMP图片*/
    FILE *fp=fopen(argv[1],"rb");
    if(fp==NULL)
    {
        printf("%s 文件不存在.\n",argv[1]);
        return 0;
    }
    /*2. 读取BMP的文件头*/
    int cnt;
    struct _BMP_HEAD bmp_head;
    cnt=fread(&bmp_head,1,sizeof(struct _BMP_HEAD),fp);
    printf("成功读取:%d 字节.\n",cnt);
    printf("图片类型:%c%c\n",bmp_head.type[0],bmp_head.type[1]);
    printf("文件大小:%d\n",bmp_head.size);
    printf("数据距离文件头的偏移量:%d\n",bmp_head.seek);
    /*3. 读取文件参数信息*/
    struct _BMP_INFO bmp_info;
    cnt=fread(&bmp_info,1,sizeof(struct _BMP_INFO),fp);
    printf("成功读取:%d 字节.\n",cnt);
    printf("当前结构体大小:%d\n",bmp_info.size);
    printf("当前图片宽度:%d\n",bmp_info.w);
    printf("当前图片高度:%d\n",bmp_info.h);
    printf("当前图片颜色位数:%d\n",bmp_info.bit);
    printf("当前图片的压缩情况:%d\n",bmp_info.r1);
    /*4. 关闭文件*/
    fclose(fp);
    return 0;
}

2.2 创建一张纯色图片

#include <stdio.h>
#include <string.h>
#include <stdlib.h>

#pragma pack(1) //强制1个字节对齐
//BMP的文件头
struct _BMP_HEAD
{
    char type[2]; //图片的类型 "BM"
    unsigned int size; //文件大小
    unsigned short  r1; //保留1
    unsigned short  r2; //保留2
    unsigned int seek; //数据偏移字节(真实像素点数据)
};

//BMP的参数信息
struct _BMP_INFO
{
    unsigned int size; //当前结构体大小
    unsigned int w; //宽度
    unsigned int h; //高度
    unsigned short flag; //固定为1
    unsigned short bit; //像素点的位数
    unsigned int r1; //压缩方式  0
    unsigned int r2; //水平分辨率
    unsigned int r3; //垂直分辨率
    unsigned int r4; //垂直分辨率
    unsigned int r5; //引用色彩
    unsigned int r6; //关键色彩
};

int main(int argc,char **argv)
{
    if(argc!=2)
    {
        printf("传入的参数格式: ./a.out <新图片的名称>\n");
        return 0;
    }
    
    /*1. 创建一张BMP图片*/
    FILE *fp=fopen(argv[1],"wb");
    if(fp==NULL)
    {
        printf("%s 文件创建失败.\n",argv[1]);
        return 0;
    }
    /*2. 创建BMP的文件头*/
    int cnt;
    struct _BMP_HEAD bmp_head;
    memset(&bmp_head,0,sizeof(struct _BMP_HEAD));
    //图片的类型
    bmp_head.type[0]='B';
    bmp_head.type[1]='M';
    //文件大小
    bmp_head.size=54+800*480*3;
    //数据偏移量
    bmp_head.seek=54;
    //写文件头
    cnt=fwrite(&bmp_head,1,sizeof(struct _BMP_HEAD),fp);
    printf("成功写入:%d 字节.\n",cnt);

    /*3. 写文件参数信息*/
    struct _BMP_INFO bmp_info;
    memset(&bmp_info,0,sizeof(struct _BMP_INFO));
    //当前结构体大小
    bmp_info.size=sizeof(struct _BMP_INFO);
    //图片的宽度和高度
    bmp_info.w=800;
    bmp_info.h=480;
    //图片的颜色位数
    bmp_info.bit=24;
    //标志位
    bmp_info.flag=1;
    //写入文件参数信息
    cnt=fwrite(&bmp_info,1,sizeof(struct _BMP_INFO),fp);
    printf("成功写入:%d 字节.\n",cnt);

    /*4. 写入位图数据*/
    int w,h;
    int c=0xFF0033; //红色
    for(h=0;h<480;h++)
    {
        for(w=0;w<800;w++)
        {
            fwrite(&c,1,3,fp);
        }
    }
    /*5. 关闭文件*/
    fclose(fp);
    return 0;
}
目录
相关文章
|
11天前
|
存储 C语言
`scanf`是C语言中用于按格式读取标准输入的函数
`scanf`是C语言中用于按格式读取标准输入的函数,通过格式字符串解析输入并存入指定变量。需注意输入格式严格匹配,并建议检查返回值以确保读取成功,提升程序健壮性。
421 0
|
7月前
|
算法 PyTorch 算法框架/工具
昇腾 msmodelslim w8a8量化代码解析
msmodelslim w8a8量化算法原理和代码解析
459 5
|
9月前
|
搜索推荐 UED Python
实现一个带有昼夜背景切换的动态时钟:从代码到功能解析
本文介绍了一个使用Python和Tkinter库实现的动态时钟程序,具有昼夜背景切换、指针颜色随机变化及整点和半点报时功能。通过设置不同的背景颜色和随机变换指针颜色,增强视觉吸引力;利用多线程技术确保音频播放不影响主程序运行。该程序结合了Tkinter、Pygame、Pytz等库,提供了一个美观且实用的时间显示工具。欢迎点赞、关注、转发、收藏!
381 94
|
7月前
|
人工智能 小程序 前端开发
【一步步开发AI运动小程序】十九、运动识别中如何解析RGBA帧图片?
本文介绍了如何将相机抽取的RGBA帧图像解析为`.jpg`或`.png`格式,适用于体测、赛事等场景。首先讲解了RGBA图像结构,其为一维数组,每四个元素表示一个像素的颜色与透明度值。接着通过`uni.createOffscreenCanvas()`创建离屏画布以减少绘制干扰,并提供代码实现,将RGBA数据逐像素绘制到画布上生成图片。最后说明了为何不直接使用拍照API及图像转换的调用频率建议,强调应先暂存帧数据,运动结束后再进行转换和上传,以优化性能。
|
7月前
|
传感器 监控 Java
Java代码结构解析:类、方法、主函数(1分钟解剖室)
### Java代码结构简介 掌握Java代码结构如同拥有程序世界的建筑蓝图,类、方法和主函数构成“黄金三角”。类是独立的容器,承载成员变量和方法;方法实现特定功能,参数控制输入环境;主函数是程序入口。常见错误包括类名与文件名不匹配、忘记static修饰符和花括号未闭合。通过实战案例学习电商系统、游戏角色控制和物联网设备监控,理解类的作用、方法类型和主函数任务,避免典型错误,逐步提升编程能力。 **脑图速记法**:类如太空站,方法即舱段;main是发射台,static不能换;文件名对仗,括号要成双;参数是坐标,void不返航。
254 5
|
9月前
|
JSON 前端开发 搜索推荐
关于商品详情 API 接口 JSON 格式返回数据解析的示例
本文介绍商品详情API接口返回的JSON数据解析。最外层为`product`对象,包含商品基本信息(如id、name、price)、分类信息(category)、图片(images)、属性(attributes)、用户评价(reviews)、库存(stock)和卖家信息(seller)。每个字段详细描述了商品的不同方面,帮助开发者准确提取和展示数据。具体结构和字段含义需结合实际业务需求和API文档理解。
|
8月前
|
人工智能 文字识别 自然语言处理
保单AI识别技术及代码示例解析
车险保单包含基础信息、车辆信息、人员信息、保险条款及特别约定等关键内容。AI识别技术通过OCR、文档结构化解析和数据校验,实现对保单信息的精准提取。然而,版式多样性、信息复杂性、图像质量和法律术语解析是主要挑战。Python代码示例展示了如何使用PaddleOCR进行保单信息抽取,并提出了定制化训练、版式分析等优化方向。典型应用场景包括智能录入、快速核保、理赔自动化等。未来将向多模态融合、自适应学习和跨区域兼容性发展。
|
10月前
|
人工智能 搜索推荐 API
Cobalt:开源的流媒体下载工具,支持解析和下载全平台的视频、音频和图片,支持多种视频质量和格式,自动提取视频字幕
cobalt 是一款开源的流媒体下载工具,支持全平台视频、音频和图片下载,提供纯净、简洁无广告的体验
1417 9
Cobalt:开源的流媒体下载工具,支持解析和下载全平台的视频、音频和图片,支持多种视频质量和格式,自动提取视频字幕
|
8月前
|
数据采集 Web App开发 JavaScript
DOMParser解析TikTok页面中的图片元素
DOMParser解析TikTok页面中的图片元素
|
10月前
|
自然语言处理 搜索推荐 数据安全/隐私保护
鸿蒙登录页面好看的样式设计-HarmonyOS应用开发实战与ArkTS代码解析【HarmonyOS 5.0(Next)】
鸿蒙登录页面设计展示了 HarmonyOS 5.0(Next)的未来美学理念,结合科技与艺术,为用户带来视觉盛宴。该页面使用 ArkTS 开发,支持个性化定制和无缝智能设备连接。代码解析涵盖了声明式 UI、状态管理、事件处理及路由导航等关键概念,帮助开发者快速上手 HarmonyOS 应用开发。通过这段代码,开发者可以了解如何构建交互式界面并实现跨设备协同工作,推动智能生态的发展。
537 10
鸿蒙登录页面好看的样式设计-HarmonyOS应用开发实战与ArkTS代码解析【HarmonyOS 5.0(Next)】

推荐镜像

更多
  • DNS