BMP位图原理深度解析及编程实现RGB565图片格式转换

1、前言

        在Windows的画图软件中可以看到,常见的BMP有如下图所示的几种:单色位图16色位图256色位图24位位图,其颜色深度分别为1、4、8、24。

        在一些单片机设备中的LCD显示屏幕中,仅仅支持RGB565这一类的16位颜色深度图像,否则图片显示会有异常。但是在Windows中,并没有直接提供16位颜色深度的BMP图片,需要通过特殊的方式去生成。对于非专业人士,可以使用如PS、GIMP等软件导出RBG565格式的16色位BMP位图。

        但为了生成16位的BMP位图去下载大型的图像处理软件比较费时,对于有程序代码经验的开发者,往往使用编程等方式会更加的方便。因此,本文除了对BMP图片进行讲解外,还将以C语言编程的方式,实现RBG565格式的BMP位图生成转换

2、BMP位图解析

(1)、BMP基础铺垫

        BMP:BMP(位图)是一种没有经过压缩的图像格式。图片的大小取决于图像的分辨率和颜色深度。例如,一个分辨率为1024x768像素,颜色深度为24位(每个像素8位红色、8位绿色、8位蓝色)的BMP图像,其大小约为:1024 * 768 * 3 = 2,359,296字节 ≈ 2.3MB。

        BMP位图因为没有任何的压缩,因此文件尺寸都比较大,不适合在互联网上传播,优点是数据读取出来即可使用,无需任何解码器支持

        虽然BMP格式文件内部存储的就是RGB数据,无需任何解码,但毕竟RGB数据是纯数据,没有任何图片尺寸、色深等具体信息,因此我们需要了解BMP的格式头,在BMP格式头中获取图片的相关信息,然后才能正确处理内涵的RGB数据。

        另外通过研究BMP格式会发现,其RGB的存储遵从一定的规则,比如上下颠倒(扫描方式是从下往上)、4字节行距等,这些在编写程序代码时,都必须要弄清楚,否则图片不能正常显示。

(2)、BMP 格式头解析

        BMP文件开头部分是BMP格式头,里面存放了RGB数据的尺寸、分辨率、色深等重要信息。

        BMP格式头中包含了如下三个结构体:

  • bitmap_header(必有)
  • bitmap_info(必有)
  • rgb_quad(可选,一般没有)

        BMP位图结构体具体定义如下:

struct bitmap_header  // 文件头
{int16_t type; //位图文件的类型,必须为BM(1-2字节)int32_t size; //位图文件的大小,以字节为单位(3-6字节,低位在前)int16_t reserved1;//位图文件保留字,必须为0(7-8字节)int16_t reserved2;//位图文件保留字,必须为0(9-10字节)int32_t offbits; //位图数据的起始位置,以相对于位图(11-14字节,低位在前),文件头的偏移量表示,以字节为单位
}__attribute__((packed));  // 14 字节struct bitmap_info   // 信息头
{int32_t size;   //本结构所占用字节数(15-18字节)int32_t width;  //位图的宽度,以像素为单位(19-22字节)int32_t height; //位图的高度,以像素为单位(23-26字节)int16_t planes; //目标设备的级别,必须为1(27-28字节)int16_t bit_count; //每个像素所需的位数,必须是1(双色),(29-30字节)//4(16色),8(256色)16(高彩色)或24(真彩色)之一int32_t compression;//位图压缩类型,必须是0(不压缩),(31-34字节) //1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一int32_t size_img; //位图的大小(其中包含了为了补齐列数是4的倍数而添加的空字节),以字节为单位(35-38字节)int32_t X_pel;    //位图水平分辨率,每米像素数(39-42字节)int32_t Y_pel;    //位图垂直分辨率,每米像素数(43-46字节)int32_t clrused;  //位图实际使用的颜色表中的颜色数(47-50字节)int32_t clrImportant; //位图显示过程中重要的颜色数(51-54字节)
}__attribute__((packed));// 以下结构体不一定存在于BMP文件中,除非:
// bitmap_info.compression为真
struct rgb_quad
{int8_t blue;    //蓝色的亮度(值范围为0-255)int8_t green;   //绿色的亮度(值范围为0-255)int8_t red;     //红色的亮度(值范围为0-255)int8_t reserved;//保留,必须为0
}__attribute__((packed));

(3)、BMP位图数据体

        BMP位图中,这部分的内容根据BMP位图使用的位数不同而不同,在24位图中直接使用RGB,而其他的小于24位的使用调色板中颜色索引值

        位图数据记录了位图的每一个像素值,记录顺序是在扫描行内是从左到右,扫描行间是从下到上

        位图的一个像素值所占的字节数:

当bit_count = 1时,8个像素占1个字节;

当bit_count = 4时,2个像素占1个字节;

当bit_count = 8时,1个像素占1个字节;

当bit_count = 24时,1个像素占3个字节,按顺序分别为B,G,R;

3、RGB565转换代码

(1)、bmp_convert.h

#ifndef __BMP_CONVERT
#define __BMP_CONVERT#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>typedef struct bitmap_header  // BMP文件头
{int16_t type;       //文件的类型,必须为BM(1-2字节)int32_t size;       //文件的大小,字节为单位(3-6字节,低位在前)int16_t reserved1;  //文件保留字,必须为0(7-8字节)int16_t reserved2;  //文件保留字,必须为0(9-10字节)int32_t offbits;    //文件头的偏移量,以字节为单位(11-14字节,低位在前)
}__attribute__((packed))BMPHeader;  // 14 字节typedef struct bitmap_info   // BMP信息头
{int32_t size;   //本结构所占用字节数(15-18字节)int32_t width;  //位图的宽度,以像素为单位(19-22字节)int32_t height; //位图的高度,以像素为单位(23-26字节)int16_t planes; //目标设备的级别,必须为1(27-28字节)int16_t bit_count;  //每个像素所需的位数(29-30字节)//4(16色),8(256色)16(高彩色)或24(真彩色)之一int32_t compression;//位图压缩类型,必须是0(不压缩),(31-34字节) //1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一int32_t size_img;   //位图的大小(其中包含了为了补齐列数是4的倍数而添加的空字节),以字节为单位(35-38字节)int32_t X_pel;      //位图水平分辨率,每米像素数(39-42字节)int32_t Y_pel;      //位图垂直分辨率,每米像素数(43-46字节)int32_t clrused;    //位图实际使用的颜色表中的颜色数(47-50字节)int32_t clrImportant; //位图显示过程中重要的颜色数(51-54字节)
}__attribute__((packed))BMPInfoHeader;// 以下结构体不一定存在于BMP文件中,除非:
// bitmap_info.compression为真
typedef struct rgb_quad 
{int8_t blue;    //蓝色的亮度(值范围为0-255)int8_t green;   //绿色的亮度(值范围为0-255)int8_t red;     //红色的亮度(值范围为0-255)int8_t reserved;//保留,必须为0
}__attribute__((packed))BMPRgb;typedef struct {unsigned char blue;unsigned char green;unsigned char red;
} RGB24;typedef struct {unsigned short blue : 5;unsigned short green : 6;unsigned short red : 5;
} RGB16;void rgb24_to_rgb16(RGB24 *src, RGB16 *dst, int numPixels);
int bmp24_to_bmp16(char *file_src, char *file_dst);#endif

        在 C 语言中,结构体或联合体的成员通常会按照其自然对齐(natural alignment)进行排列。这意味着编译器会在成员之间插入填充字节,以确保每个成员都位于其对齐要求的边界上。这种对齐可以提高内存访问速度,但也可能导致结构体或联合体占用更多的内存空间

        使用 __attribute__((packed)) 可以避免这种默认的内存填充,使得结构体或联合体的成员紧密排列。这在处理硬件寄存器或其他需要紧凑布局的场景中非常有用。

(2)、bmp_convert.c

#include "bmp_convert.h"void rgb24_to_rgb16(RGB24 *src, RGB16 *dst, int numPixels)
{for (int i = 0; i < numPixels; i++) {dst[i].red = (src[i].red >> 3) & 0x1F;dst[i].green = (src[i].green >> 2) & 0x3F;dst[i].blue = (src[i].blue >> 3) & 0x1F;}
}int bmp24_to_bmp16(char *file_src, char *file_dst)
{FILE *infile = fopen(file_src, "rb");if (!infile) {perror("Error opening input file");return -1;}BMPHeader header;fread(&header, sizeof(header), 1, infile);if (header.type != 0x4D42) {printf("Invalid BMP file\n");return -1;}BMPInfoHeader infoHeader;fread(&infoHeader, sizeof(infoHeader), 1, infile);printf("biBit:%d\n", infoHeader.bit_count);if (infoHeader.bit_count != 24) {printf("Input file is not 24-bit BMP\n");fclose(infile);return -1;}fseek(infile, header.offbits, SEEK_SET);int width = infoHeader.width;int height = infoHeader.height;int numPixels = width * height;RGB24 *rgb24 = (RGB24 *)malloc(numPixels * sizeof(RGB24));fread(rgb24, sizeof(RGB24), numPixels, infile);fclose(infile);RGB16 *rgb16 = (RGB16 *)malloc(numPixels * sizeof(RGB16));rgb24_to_rgb16(rgb24, rgb16, numPixels);FILE *outfile = fopen(file_dst, "wb");if (!outfile) {perror("Error opening output file");free(rgb24);free(rgb16);return -1;}fwrite(&header, sizeof(header), 1, outfile);infoHeader.bit_count = 16;fwrite(&infoHeader, sizeof(infoHeader), 1, outfile);fwrite(rgb16, sizeof(RGB16), numPixels, outfile);fclose(outfile);free(rgb24);free(rgb16);return 0;
}

(3)、程序效果

        如下图所示为24位颜色深度的BMP原图和经过转换后的16位颜色深度BMP图。转换后,肉眼可见的图像部分颜色丢失、且存储空间变小。图片放到支持RGB565格式的显示设备中,转换后的位图效果将会效果更佳。如果需要转换后的图片数据失真较少,需要额外特点的图片处理算法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/409755.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[windows][软件]Windows平台MongoDB的安装

1.下载软件 上mongoDB官网&#xff0c;网址&#xff1a;Download MongoDB Community Server | MongoDB&#xff0c; 下载对应的版本软件 2.安装 下载安装包如图&#xff1a; 双击安装&#xff1a; 默认&#xff0c;点击next 默认,点击next 默认点Complete&#xff0c;完整安…

混杂设备驱动、Linux内核中的中断、火焰传感器驱动、呼吸传感器驱动、等待队列

混杂设备驱动 混杂设备也叫杂项设备&#xff0c;是对普通的字符设备(struct cdev)的一种封装。misc 设备会自动创建cdev&#xff0c;不需要像我 们以前那样手动创建&#xff0c;因此采用misc 设备驱动可以简化字符设备驱动的编写。具有以下特点&#xff1a; 1) 主设备号为10&…

DVWA靶场通关(CSRF)

CSRF 是跨站请求伪造&#xff0c;是指利用受害者尚未失效的身份认证信息&#xff08;cookie、会话等&#xff09;&#xff0c;诱骗其点击恶意链接或者访问包含攻击代码的页面&#xff0c;在受害人不知情的情况下以受害者的身份向&#xff08;身份认证信息所对应的&#xff09;服…

WRF-LES与PALM微尺度气象大涡模拟

针对微尺度气象的复杂性&#xff0c;大涡模拟&#xff08;LES&#xff09;提供了一种无可比拟的解决方案。微尺度气象学涉及对小范围内的大气过程进行精确模拟&#xff0c;这些过程往往与天气模式、地形影响和人为因素如城市布局紧密相关。在这种规模上&#xff0c;传统的气象模…

生信学院|09月06日《在线产品交流工具》

课程主题&#xff1a;在线产品交流工具 课程时间&#xff1a;2024年09月06日 14:00-14:30 主讲人&#xff1a;曾裕章 生信科技 售后服务工程师 3DEXPERIENCE云平台基于云平台的交流工具XPR功能讲解Q&A 安装腾讯会议客户端或APP报名哦~~~ 或者点击链接报名&#xff1a;…

C语言进阶(一)数据在内存中的存储

整数在内存中的存储 整数的2进制表示方法有三种&#xff0c;即原码、反码和补码 有符号的整数&#xff0c;三种表示方法均有符号位和数值位两部分 符号位用0表示“正”&#xff0c;用1表示“负”&#xff0c;最高位被当做符号位&#xff0c;剩余的都是数值位 正整数的原、反…

构建buildroot根文件系统

目录 1.确定gcc工具版本2.下载Buildroot源码并编译2.1 下载Buildroot源码2.2 配置Buildroot2.2.1 配置 Target options2.2.2 配置交叉编译工具链2.2.3 配置 System configuration2.2.4 配置 Filesystem images2.2.5 禁止编译 Linux 内核和 uboot2.2.6 编译Buildroot源码2.2.7 查…

【多线程】深入剖析线程安全问题

&#x1f490;个人主页&#xff1a;初晴~ &#x1f4da;相关专栏&#xff1a;多线程 / javaEE初阶 前言 线程安全问题是在多线程学习中一个十分重要的话题。多个线程并发执行就容易产生许多冲突与问题&#xff0c;如何协调好每个线程的执行&#xff0c;让多线程编程“多而不乱…

【Node】【3】回调函数

nodejs 是一个基于事件驱动和非阻塞异步的JavaScript运行时环境。 Node.js 采用单线程模型&#xff0c; 单线程意味着 Node.js 在任何给定时刻只能执行一段代码&#xff0c;但通过异步执行回调函数&#xff0c;可以在等待 I/O 操作完成的同时继续执行其他代码&#xff0c;从而…

每日一练-threejs实现三维动态热力图

前言&#xff1a;学习自用Three.js搞个炫酷热力山丘图&#xff0c;作者讲解的十分详细&#xff0c;在这里不再过多赘述&#xff0c;直接上代码&#xff01; <template><div class"map" ref"map"></div> </template><script set…

XTuner微调个人小助手认知 #书生浦语大模型实战营#

1.任务&#xff1a; 本次的任务是使用 XTuner 微调 InternLM2-Chat-1.8B 实现自己的小助手认知&#xff0c;从而让模型能够个性化的回复&#xff0c;让模型知道他是我们的小助手&#xff0c;在实战营帮我们完成XTuner微调个人小助手认知的任务。并截图打卡。 任务打卡&#x…

书生.浦江大模型实战训练营——(十一)LMDeploy 量化部署进阶实践

最近在学习书生.浦江大模型实战训练营&#xff0c;所有课程都免费&#xff0c;以关卡的形式学习&#xff0c;也比较有意思&#xff0c;提供免费的算力实战&#xff0c;真的很不错&#xff08;无广&#xff09;&#xff01;欢迎大家一起学习&#xff0c;打开LLM探索大门&#xf…

复杂的编辑表格

需求描述 表格可以整体编辑&#xff1b;也可以单行弹框编辑&#xff1b;且整体编辑的时候&#xff0c;依然可以单行编辑 编辑只能给某一列&#xff08;这里是参数运行值&#xff09;修改&#xff0c;且根据数据内容的参数范围来判断展示不同的形式&#xff1a;input/数字输入/单…

计算机网络——TCP协议与UDP协议详解(下)

一、TCP协议 1.1 TCP协议的报文 TCP全称为 "传输控制协议(Transmission Control Protocol")。人如其名&#xff0c;要对数据的传输进行一个详细的控制。我们先看其报文格式&#xff0c;如下图&#xff1a; TCP报文由以下几个字段组成&#xff1a; 源端口号和目标端口…

MySQL索引详解:原理、数据结构与分析和优化

在数据库管理系统中&#xff0c;索引是提高查询性能、优化数据存储结构的重要工具。MySQL作为广泛使用的开源关系型数据库管理系统&#xff0c;其索引机制对于提升数据库操作效率具有至关重要的作用。本文将围绕“MySQL索引详解&#xff1a;原理、数据结构与分析和优化”这一主…

CRUD的最佳实践,联动前后端,包含微信小程序,API,HTML等(二)

CRUD老生常谈&#xff0c;但是我搜索了一圈&#xff0c;发觉几乎是着重在后端&#xff0c;也就是API部分&#xff01; 无外乎2个思路 1.归总的接口&#xff0c;比如一个接口&#xff0c;实现不同表的CRUD 2.基于各自的表&#xff0c;使用代码生成器实现CRUD 个人来说是推荐2&am…

Harmony鸿蒙应用开发:解决Web组件加载本地资源跨域

鸿蒙开发文档中有一节 加载本地页面 提到了可以通过 $rawfile 方法加载本地 HTML 网页&#xff1a; Index.ets 1Web({ src: $rawfile("local.html"), controller: this.webviewController })但是如果在 local.html 中需要引用一些静态资源&#xff0c;例如图片、JS、…

MMS论文中关于语种识别的内容摘要

MMS论文中关于语种识别的内容摘要 前言语种识别相关内容实验结论 前言 摘要翻译一些内容。 论文地址请看这里 语种识别相关内容 Whisper支持LID&#xff0c;可以区分99种不同的语言&#xff1b;有人使用wav2vec 2.0实现LID&#xff0c;数据集中包含10种亚洲语言&#xff1b;…

JavaScript - Ajax

Asynchronous JavaScript And XML&#xff0c;异步的JavaScript和XML 作用: 数据交换&#xff1a;通过Ajax可以给服务器发送请求&#xff0c;并获取服务器响应的数据。异步交互&#xff1a;可以在不重新加载整个页面的情况下&#xff0c;与服务器交换数据并更新部分网页的技术…

[新手入门]1台电脑+1个电视+2个软件(sunshine+moonlight) 解决黑神话悟空没有hdmi线的痛...

sunshinemoonlight 解决黑神话悟空 本地串流投屏 背景:偶然间在B站发现了sunshinemoonlight方案,替代hdmi线,做本地串流...于是心灵手巧的我开始尝试踩坑之路:1.准备安装包2.开始安装2.1 笔记本windows安装sunshine2.2 遇到了第一个坑.Fatal: ViGEmBus is not installed or run…