c语言代码实现位图直方图均衡,编写一个程序,完成指定文件的直方图均衡,并存储到新文件,file1.bmp file2.bmp

时间: 2024-03-08 13:49:15 浏览: 20
好的,下面是实现代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #pragma pack(push, 1) typedef struct { unsigned short bfType; // 位图文件的类型,必须为BM unsigned int bfSize; // 位图文件的大小,以字节为单位 unsigned short bfReserved1; // 位图文件保留字,必须为0 unsigned short bfReserved2; // 位图文件保留字,必须为0 unsigned int bfOffBits; // 位图数据的起始位置,以相对于位图文件头的偏移量表示,以字节为单位 } BITMAPFILEHEADER; typedef struct { unsigned int biSize; // 本结构所占用字节数 int biWidth; // 位图的宽度,以像素为单位 int biHeight; // 位图的高度,以像素为单位 unsigned short biPlanes; // 目标设备的级别,必须为1 unsigned short biBitCount; // 每个像素所需的位数,必须是1、4、8或24 unsigned int biCompression; // 位图压缩类型,必须是0(不压缩)、1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型) unsigned int biSizeImage; // 位图的大小,以字节为单位 int biXPelsPerMeter; // 位图水平分辨率,每米像素数 int biYPelsPerMeter; // 位图垂直分辨率,每米像素数 unsigned int biClrUsed; // 位图实际使用的颜色表中的颜色数 unsigned int biClrImportant; // 位图显示过程中重要的颜色数 } BITMAPINFOHEADER; #pragma pack(pop) typedef struct { unsigned char b, g, r; } RGB; // 计算直方图 void calc_histogram(unsigned char *data, int width, int height, int *hist) { memset(hist, 0, sizeof(int) * 256); for (int i = 0; i < height; i++) { for (int j = 0; j < width; j++) { hist[data[i * width + j]]++; } } } // 直方图均衡 void histogram_equalization(unsigned char *src, int width, int height, unsigned char *dst) { int hist[256]; calc_histogram(src, width, height, hist); int cum_hist[256]; for (int i = 0; i < 256; i++) { if (i == 0) { cum_hist[i] = hist[i]; } else { cum_hist[i] = cum_hist[i - 1] + hist[i]; } } int max_val = width * height; for (int i = 0; i < height; i++) { for (int j = 0; j < width; j++) { int val = src[i * width + j]; dst[i * width + j] = (unsigned char)(cum_hist[val] * 255 / max_val); } } } // 读取位图文件 RGB *read_bitmap(const char *filename, int *width, int *height) { FILE *fp = fopen(filename, "rb"); if (!fp) { return NULL; } BITMAPFILEHEADER fh; BITMAPINFOHEADER ih; fread(&fh, sizeof(BITMAPFILEHEADER), 1, fp); fread(&ih, sizeof(BITMAPINFOHEADER), 1, fp); *width = ih.biWidth; *height = ih.biHeight; int line_bytes = ((*width * ih.biBitCount + 31) / 32) * 4; RGB *data = (RGB *)malloc(line_bytes * (*height)); fseek(fp, fh.bfOffBits, SEEK_SET); for (int i = 0; i < *height; i++) { fread(&data[(i * line_bytes) / 3], sizeof(RGB), *width, fp); fseek(fp, line_bytes - (*width * 3), SEEK_CUR); } fclose(fp); return data; } // 写入位图文件 void write_bitmap(const char *filename, RGB *data, int width, int height) { FILE *fp = fopen(filename, "wb"); if (!fp) { return; } int line_bytes = ((width * 24 + 31) / 32) * 4; BITMAPFILEHEADER fh = { 'B' + ('M' << 8), line_bytes * height + sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER), 0, 0, sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) }; BITMAPINFOHEADER ih = { sizeof(BITMAPINFOHEADER), width, height, 1, 24, 0, line_bytes * height, 0, 0, 0, 0 }; fwrite(&fh, sizeof(BITMAPFILEHEADER), 1, fp); fwrite(&ih, sizeof(BITMAPINFOHEADER), 1, fp); for (int i = 0; i < height; i++) { fwrite(&data[(height - i - 1) * width], sizeof(RGB), width, fp); fwrite("\0\0\0", 1, line_bytes - width * 3, fp); } fclose(fp); } int main() { int width, height; RGB *data = read_bitmap("file1.bmp", &width, &height); unsigned char *gray_data = (unsigned char *)malloc(width * height); for (int i = 0; i < height; i++) { for (int j = 0; j < width; j++) { RGB *pixel = &data[i * width + j]; gray_data[i * width + j] = (unsigned char)(0.299 * pixel->r + 0.587 * pixel->g + 0.114 * pixel->b); } } unsigned char *eq_data = (unsigned char *)malloc(width * height); histogram_equalization(gray_data, width, height, eq_data); RGB *eq_rgb_data = (RGB *)malloc(width * height * sizeof(RGB)); for (int i = 0; i < height; i++) { for (int j = 0; j < width; j++) { unsigned char val = eq_data[i * width + j]; eq_rgb_data[i * width + j].r = val; eq_rgb_data[i * width + j].g = val; eq_rgb_data[i * width + j].b = val; } } write_bitmap("file2.bmp", eq_rgb_data, width, height); free(data); free(gray_data); free(eq_data); free(eq_rgb_data); return 0; } ``` 这个程序会读取指定的位图文件,将其转换为灰度图像,然后进行直方图均衡,最后将均衡后的图像存储到新文件中。程序中使用的是 24 位真彩色位图,如果需要处理其他格式的位图,需要修改代码中写入和读取位图的部分。

相关推荐

最新推荐

recommend-type

c语言实现把文件中数据读取并存到数组中

下面小编就为大家带来一篇c语言实现把文件中数据读取并存到数组中。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
recommend-type

利用C语言替换文件中某一行的方法

大家都知道C语言提供了文件操作,但是替换文件的某一行比较麻烦,下面是我使用的一个方法,现在分享给大家,有需要的朋友们可以参考借鉴。
recommend-type

怎么使用Keil C51建立一个新项目并添加C语言程序文件

怎么使用Keil C51建立一个新项目并添加C语言程序文件 很详细简洁的说明
recommend-type

C语言从txt文件中逐行读入数据存到数组中的实现方法

下面小编就为大家带来一篇C语言从txt文件中逐行读入数据存到数组中的实现方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
recommend-type

毕业设计MATLAB_执行一维相同大小矩阵的QR分解.zip

毕业设计matlab
recommend-type

zigbee-cluster-library-specification

最新的zigbee-cluster-library-specification说明文档。
recommend-type

管理建模和仿真的文件

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

MATLAB柱状图在信号处理中的应用:可视化信号特征和频谱分析

![matlab画柱状图](https://img-blog.csdnimg.cn/3f32348f1c9c4481a6f5931993732f97.png) # 1. MATLAB柱状图概述** MATLAB柱状图是一种图形化工具,用于可视化数据中不同类别或组的分布情况。它通过绘制垂直条形来表示每个类别或组中的数据值。柱状图在信号处理中广泛用于可视化信号特征和进行频谱分析。 柱状图的优点在于其简单易懂,能够直观地展示数据分布。在信号处理中,柱状图可以帮助工程师识别信号中的模式、趋势和异常情况,从而为信号分析和处理提供有价值的见解。 # 2. 柱状图在信号处理中的应用 柱状图在信号处理
recommend-type

帮我设计一个基于Android平台的便签APP的代码

很抱歉,我是一个语言模型AI,无法提供完整的Android APP代码。但是,我可以给您一些设计便签APP的建议和思路。 1. 界面设计 首先考虑APP的界面设计,便签APP通常需要一个主界面用于显示所有的便签,以及一个编辑界面用于编辑单个便签。主界面可以采用列表的形式,列出所有的便签,每个便签可以显示标题和摘要内容。在编辑界面中,用户可以输入标题和正文内容,并且可以设置提醒时间、标签、优先级等。 2. 数据存储 便签APP需要一个数据存储的方案,可以考虑使用SQLite数据库来存储便签数据。每个便签可以存储标题、正文内容、提醒时间、标签、优先级等信息。 3. 便签操作 便签APP
recommend-type

JSBSim Reference Manual

JSBSim参考手册,其中包含JSBSim简介,JSBSim配置文件xml的编写语法,编程手册以及一些应用实例等。其中有部分内容还没有写完,估计有生之年很难看到完整版了,但是内容还是很有参考价值的。