如何用C++代码实现BMP图片叠加

时间: 2023-07-28 07:06:57 浏览: 63
BMP 图片叠加的 C++ 实现可以参考以下代码: ```c++ #include <iostream> #include <fstream> #include <vector> #include <cstring> using namespace std; #pragma pack(push, 1) // BMP 文件头结构体 typedef struct tagBITMAPFILEHEADER { unsigned short bfType; // 文件类型,必须为 "BM" unsigned int bfSize; // 文件大小,单位为字节 unsigned short bfReserved1; // 保留字段 unsigned short bfReserved2; // 保留字段 unsigned int bfOffBits; // 从文件头到像素数据的偏移量,单位为字节 } BITMAPFILEHEADER; // BMP 信息头结构体 typedef struct tagBITMAPINFOHEADER { unsigned int biSize; // 信息头大小,单位为字节 int biWidth; // 图像宽度,单位为像素 int biHeight; // 图像高度,单位为像素 unsigned short biPlanes; // 位平面数,必须为 1 unsigned short biBitCount; // 每像素位数,常用的为 1、4、8、24 unsigned int biCompression; // 压缩方式,常用的为 0(不压缩) unsigned int biSizeImage; // 像素数据大小,单位为字节 int biXPelsPerMeter; // 水平分辨率,单位为像素/米 int biYPelsPerMeter; // 垂直分辨率,单位为像素/米 unsigned int biClrUsed; // 使用的颜色数,如果为 0,则使用所有颜色 unsigned int biClrImportant; // 重要的颜色数,如果为 0,则都重要 } BITMAPINFOHEADER; // BMP 调色板结构体 typedef struct tagRGBQUAD { unsigned char rgbBlue; // 蓝色分量 unsigned char rgbGreen; // 绿色分量 unsigned char rgbRed; // 红色分量 unsigned char rgbReserved; // 保留字段 } RGBQUAD; #pragma pack(pop) // 读取 BMP 文件 bool loadBmp(const char *filename, vector<unsigned char> &imageData, BITMAPFILEHEADER &bmpFileHeader, BITMAPINFOHEADER &bmpInfoHeader, vector<RGBQUAD> &bmpPalette) { // 打开 BMP 文件 ifstream file(filename, ios::binary); if (!file) { cerr << "Error: BMP file not found." << endl; return false; } // 读取 BMP 文件头 file.read(reinterpret_cast<char *>(&bmpFileHeader), sizeof(BITMAPFILEHEADER)); if (bmpFileHeader.bfType != 0x4D42) { cerr << "Error: Invalid BMP file format." << endl; return false; } // 读取 BMP 信息头 file.read(reinterpret_cast<char *>(&bmpInfoHeader), sizeof(BITMAPINFOHEADER)); if (bmpInfoHeader.biBitCount != 24 && bmpInfoHeader.biBitCount != 32) { cerr << "Error: Unsupported BMP bit count." << endl; return false; } // 读取 BMP 调色板(如果有) if (bmpInfoHeader.biClrUsed != 0) { bmpPalette.resize(bmpInfoHeader.biClrUsed); file.read(reinterpret_cast<char *>(bmpPalette.data()), bmpInfoHeader.biClrUsed * sizeof(RGBQUAD)); } // 读取 BMP 图像数据 imageData.resize(bmpInfoHeader.biSizeImage); file.read(reinterpret_cast<char *>(imageData.data()), bmpInfoHeader.biSizeImage); // 关闭 BMP 文件 file.close(); return true; } // 保存 BMP 文件 bool saveBmp(const char *filename, const vector<unsigned char> &imageData, const BITMAPFILEHEADER &bmpFileHeader, const BITMAPINFOHEADER &bmpInfoHeader, const vector<RGBQUAD> &bmpPalette) { // 创建 BMP 文件 ofstream file(filename, ios::binary); if (!file) { cerr << "Error: Failed to create BMP file." << endl; return false; } // 写入 BMP 文件头 file.write(reinterpret_cast<const char *>(&bmpFileHeader), sizeof(BITMAPFILEHEADER)); // 写入 BMP 信息头 file.write(reinterpret_cast<const char *>(&bmpInfoHeader), sizeof(BITMAPINFOHEADER)); // 写入 BMP 调色板(如果有) if (bmpInfoHeader.biClrUsed != 0) { file.write(reinterpret_cast<const char *>(bmpPalette.data()), bmpInfoHeader.biClrUsed * sizeof(RGBQUAD)); } // 写入 BMP 图像数据 file.write(reinterpret_cast<const char *>(imageData.data()), bmpInfoHeader.biSizeImage); // 关闭 BMP 文件 file.close(); return true; } // 将两张 BMP 图片叠加 bool blendBmp(const vector<unsigned char> &imageData1, const BITMAPINFOHEADER &bmpInfoHeader1, const vector<unsigned char> &imageData2, const BITMAPINFOHEADER &bmpInfoHeader2, vector<unsigned char> &imageData3, BITMAPINFOHEADER &bmpInfoHeader3) { // 检查两张 BMP 图片是否具有相同的尺寸和位深度 if (bmpInfoHeader1.biWidth != bmpInfoHeader2.biWidth || bmpInfoHeader1.biHeight != bmpInfoHeader2.biHeight || bmpInfoHeader1.biBitCount != bmpInfoHeader2.biBitCount) { cerr << "Error: BMP images have different sizes or bit depths." << endl; return false; } // 拷贝 BMP 信息头 memcpy(&bmpInfoHeader3, &bmpInfoHeader1, sizeof(BITMAPINFOHEADER)); // 计算 BMP 图像数据大小(注意,必须按 4 字节对齐) int rowSize = (bmpInfoHeader1.biWidth * bmpInfoHeader1.biBitCount + 31) / 32 * 4; bmpInfoHeader3.biSizeImage = rowSize * bmpInfoHeader1.biHeight; // 分配 BMP 图像数据内存 imageData3.resize(bmpInfoHeader3.biSizeImage); // 循环遍历 BMP 图像数据,并将两张 BMP 图片叠加 int imageSize = bmpInfoHeader1.biWidth * bmpInfoHeader1.biHeight * (bmpInfoHeader1.biBitCount / 8); for (int i = 0; i < imageSize; i += (bmpInfoHeader1.biBitCount / 8)) { int r1 = imageData1[i + 2]; int g1 = imageData1[i + 1]; int b1 = imageData1[i]; int r2 = imageData2[i + 2]; int g2 = imageData2[i + 1]; int b2 = imageData2[i]; int r3 = min(r1 + r2, 255); int g3 = min(g1 + g2, 255); int b3 = min(b1 + b2, 255); imageData3[i] = static_cast<unsigned char>(b3); imageData3[i + 1] = static_cast<unsigned char>(g3); imageData3[i + 2] = static_cast<unsigned char>(r3); } return true; } int main() { // 读取第一张 BMP 图片 BITMAPFILEHEADER bmpFileHeader1; BITMAPINFOHEADER bmpInfoHeader1; vector<unsigned char> imageData1; vector<RGBQUAD> bmpPalette1; if (!loadBmp("image1.bmp", imageData1, bmpFileHeader1, bmpInfoHeader1, bmpPalette1)) { return 1; } // 读取第二张 BMP 图片 BITMAPFILEHEADER bmpFileHeader2; BITMAPINFOHEADER bmpInfoHeader2; vector<unsigned char> imageData2; vector<RGBQUAD> bmpPalette2; if (!loadBmp("image2.bmp", imageData2, bmpFileHeader2, bmpInfoHeader2, bmpPalette2)) { return 1; } // 将两张 BMP 图片叠加 vector<unsigned char> imageData3; BITMAPINFOHEADER bmpInfoHeader3; if (!blendBmp(imageData1, bmpInfoHeader1, imageData2, bmpInfoHeader2, imageData3, bmpInfoHeader3)) { return 1; } // 保存叠加后的 BMP 图片 BITMAPFILEHEADER bmpFileHeader3; bmpFileHeader3.bfType = 0x4D42; bmpFileHeader3.bfSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + bmpInfoHeader3.biSizeImage; bmpFileHeader3.bfReserved1 = 0; bmpFileHeader3.bfReserved2 = 0; bmpFileHeader3.bfOffBits = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER); if (!saveBmp("image3.bmp", imageData3, bmpFileHeader3, bmpInfoHeader3, bmpPalette1)) { return 1; } return 0; } ``` 这段代码中,loadBmp() 函数用于读取 BMP 文件,saveBmp() 函数用于保存 BMP 文件,blendBmp() 函数用于将两张 BMP 图片叠加。在主函数中,首先读取两张 BMP 图片,然后将它们叠加,并保存为新的 BMP 文件。需要注意的是,这段代码中只支持 24 位和 32 位的 BMP 图片。如果需要支持其他位深度的 BMP 图片,需要进行相应的修改。

相关推荐

最新推荐

recommend-type

约瑟夫环问题用C++代码实现

8. 【题目】约瑟夫环(约瑟夫问题)是一个数学的应用问题:已知n个人(以编号1,2,3...n分别表示)围坐在一张圆桌周围。从编号为1的人开始报数,数到k的那个人出列;他的下一个人又从1开始报数,数到k的那个人又...
recommend-type

c++代码实现tea加密算法的实例详解

主要介绍了c++代码实现tea加密算法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
recommend-type

C++实现图形界面时钟表盘代码

主要介绍了C++实现图形界面时钟表盘代码,涉及坐标函数的应用及图形界面程序设计,需要的朋友可以参考下
recommend-type

用C++实现DBSCAN聚类算法

本篇文章是对使用C++实现DBSCAN聚类算法的方法进行了详细的分析介绍,需要的朋友参考下
recommend-type

生产者消费者的c++代码实现

计算机操作系统经典的生产者消费者问题c++高级语言的实现。编程入门必备。
recommend-type

RTL8188FU-Linux-v5.7.4.2-36687.20200602.tar(20765).gz

REALTEK 8188FTV 8188eus 8188etv linux驱动程序稳定版本, 支持AP,STA 以及AP+STA 共存模式。 稳定支持linux4.0以上内核。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

:YOLO目标检测算法的挑战与机遇:数据质量、计算资源与算法优化,探索未来发展方向

![:YOLO目标检测算法的挑战与机遇:数据质量、计算资源与算法优化,探索未来发展方向](https://img-blog.csdnimg.cn/7e3d12895feb4651b9748135c91e0f1a.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBA5rKJ6YaJ77yM5LqO6aOO5Lit,size_20,color_FFFFFF,t_70,g_se,x_16) # 1. YOLO目标检测算法简介 YOLO(You Only Look Once)是一种
recommend-type

def ObjFun(x,y,beta): # 目标函数 """ Logistic regression loss function :param beta: model parameter vector :param x: feature matrix :param y: label vector :return: loss value """ n = x.shape[0] p = x.shape[1] pred = 1 / (1 + np.exp(-np.dot(x, beta))) pred = np.clip(pred, 1e-15, 1 - 1e-15) # 将预测值限制在一个很小的区间内 ObjVal = -np.sum(y * np.log(pred) + (1 - y) * np.log(1 - pred)) / n return ObjVal

好的,根据上面的目标函数,我们可以推导出其对应的梯度向量函数,如下所示: def gradient(x, y, beta): """ Compute gradient of the logistic regression loss function :param beta: model parameter vector :param x: feature matrix :param y: label vector :return: gradient vector """ n = x.shape[0] pred = 1 /
recommend-type

c++校园超市商品信息管理系统课程设计说明书(含源代码) (2).pdf

校园超市商品信息管理系统课程设计旨在帮助学生深入理解程序设计的基础知识,同时锻炼他们的实际操作能力。通过设计和实现一个校园超市商品信息管理系统,学生掌握了如何利用计算机科学与技术知识解决实际问题的能力。在课程设计过程中,学生需要对超市商品和销售员的关系进行有效管理,使系统功能更全面、实用,从而提高用户体验和便利性。 学生在课程设计过程中展现了积极的学习态度和纪律,没有缺勤情况,演示过程流畅且作品具有很强的使用价值。设计报告完整详细,展现了对问题的深入思考和解决能力。在答辩环节中,学生能够自信地回答问题,展示出扎实的专业知识和逻辑思维能力。教师对学生的表现予以肯定,认为学生在课程设计中表现出色,值得称赞。 整个课程设计过程包括平时成绩、报告成绩和演示与答辩成绩三个部分,其中平时表现占比20%,报告成绩占比40%,演示与答辩成绩占比40%。通过这三个部分的综合评定,最终为学生总成绩提供参考。总评分以百分制计算,全面评估学生在课程设计中的各项表现,最终为学生提供综合评价和反馈意见。 通过校园超市商品信息管理系统课程设计,学生不仅提升了对程序设计基础知识的理解与应用能力,同时也增强了团队协作和沟通能力。这一过程旨在培养学生综合运用技术解决问题的能力,为其未来的专业发展打下坚实基础。学生在进行校园超市商品信息管理系统课程设计过程中,不仅获得了理论知识的提升,同时也锻炼了实践能力和创新思维,为其未来的职业发展奠定了坚实基础。 校园超市商品信息管理系统课程设计的目的在于促进学生对程序设计基础知识的深入理解与掌握,同时培养学生解决实际问题的能力。通过对系统功能和用户需求的全面考量,学生设计了一个实用、高效的校园超市商品信息管理系统,为用户提供了更便捷、更高效的管理和使用体验。 综上所述,校园超市商品信息管理系统课程设计是一项旨在提升学生综合能力和实践技能的重要教学活动。通过此次设计,学生不仅深化了对程序设计基础知识的理解,还培养了解决实际问题的能力和团队合作精神。这一过程将为学生未来的专业发展提供坚实基础,使其在实际工作中能够胜任更多挑战。