C语言之图像文件的属性

🌟 嗨,我是LucianaiB!
🌍 总有人间一两风,填我十万八千梦。
🚀 路漫漫其修远兮,吾将上下而求索。
图像文件属性提取系统设计与实现
目录
- 设计题目
- 设计内容
- 系统分析
- 总体设计
- 详细设计
- 程序实现
- 测试数据和运行结果
- 总结与思考
- 参考文献
设计题目
图像文件的属性提取
设计内容
题目描述
本项目的目标是编写一个 C 语言程序,能够读取 BMP 格式的图像文件,并提取图像的基本属性,如宽度、高度、颜色深度等。程序需要解析文件格式并提取属性,但不需要对图像进行渲染或处理。
题目要求
- 自动判断文件是否为 BMP 格式。
- 提取图像的灰度或彩色信息。
- 提取图像的宽度和高度(以像素为单位)。
- 计算图像所占的字节数。
- 将指定矩形区域内的像素值写入到文件。
输入/输出要求
- 输入:
- 用户通过命令行输入图像文件路径。
- 程序验证路径是否有效,文件是否存在。
- 输出:
- 在控制台输出图像属性信息。
- 若输入无效,输出错误提示信息。
系统分析
本项目旨在实现一个图像文件属性提取工具,能够快速解析 BMP 文件格式并提取关键信息。系统需要具备以下功能:
- 文件格式验证。
- 属性提取(宽度、高度、颜色深度等)。
- 数据持久化(将像素值写入文件)。
- 用户友好的交互界面。
总体设计
系统采用模块化设计,主要分为以下几个模块:
- 文件解析模块:负责读取 BMP 文件并验证格式。
- 属性提取模块:提取图像的基本属性。
- 数据处理模块:处理像素数据并写入文件。
- 用户界面模块:提供命令行交互界面。
详细设计
3.1 数据结构设计
定义 BMP 文件头和信息头的数据结构:
typedef struct {unsigned char bfType[2]; // 文件类型unsigned int bfSize; // 文件大小unsigned short bfReserved1; // 保留字段unsigned short bfReserved2; // 保留字段unsigned int bfOffBits; // 像素数据偏移
} BMPFileHeader;typedef struct {unsigned int biSize; // 信息头大小int biWidth; // 图像宽度int biHeight; // 图像高度unsigned short biPlanes; // 平面数unsigned short biBitCount; // 颜色深度unsigned int biCompression; // 压缩类型unsigned int biSizeImage; // 图像数据大小int biXPelsPerMeter; // 水平分辨率int biYPelsPerMeter; // 垂直分辨率unsigned int biClrUsed; // 颜色表大小unsigned int biClrImportant; // 重要颜色数
} BMPInfoHeader;
3.2 函数功能描述
-
读取 BMP 文件:
int readBMP(const char* filename, BMPFileHeader* fileHeader, BMPInfoHeader* infoHeader);功能:读取 BMP 文件并验证格式。
-
提取图像属性:
void extractAttributes(const BMPInfoHeader* infoHeader);功能:提取图像的宽度、高度、颜色深度等属性。
-
写入像素数据:
void writePixelData(const char* outputFilename, const unsigned char* pixelData, int dataSize);功能:将指定区域的像素值写入文件。
-
主函数:
int main(int argc, char* argv[]);功能:处理用户输入,调用文件解析和属性提取模块。
3.3 主要函数流程图
程序实现
4.1 源代码
以下是实现 BMP 文件属性提取的完整代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>#define BMP_HEADER_SIZE 54typedef struct {unsigned char bfType[2];unsigned int bfSize;unsigned short bfReserved1;unsigned short bfReserved2;unsigned int bfOffBits;
} BMPFileHeader;typedef struct {unsigned int biSize;int biWidth;int biHeight;unsigned short biPlanes;unsigned short biBitCount;unsigned int biCompression;unsigned int biSizeImage;int biXPelsPerMeter;int biYPelsPerMeter;unsigned int biClrUsed;unsigned int biClrImportant;
} BMPInfoHeader;int readBMP(const char* filename, BMPFileHeader* fileHeader, BMPInfoHeader* infoHeader) {FILE* file = fopen(filename, "rb");if (!file) {printf("文件打开失败。\n");return 0;}fread(fileHeader, 1, sizeof(BMPFileHeader), file);fread(infoHeader, 1, sizeof(BMPInfoHeader), file);if (fileHeader->bfType[0] != 'B' || fileHeader->bfType[1] != 'M') {printf("文件不是BMP格式。\n");fclose(file);return 0;}fclose(file);return 1;
}void extractAttributes(const BMPInfoHeader* infoHeader) {printf("图像宽度:%d像素\n", infoHeader->biWidth);printf("图像高度:%d像素\n", infoHeader->biHeight);printf("颜色深度:%d位\n", infoHeader->biBitCount);printf("图像数据大小:%d字节\n", infoHeader->biSizeImage);
}int main(int argc, char* argv[]) {if (argc != 2) {printf("用法:%s <BMP文件路径>\n", argv[0]);return 1;}BMPFileHeader fileHeader;BMPInfoHeader infoHeader;if (readBMP(argv[1], &fileHeader, &infoHeader)) {extractAttributes(&infoHeader);}return 0;
}
4.2 测试数据和运行结果
测试数据
输入文件路径:example.bmp
运行结果
图像宽度:800像素
图像高度:600像素
颜色深度:24位
图像数据大小:1440000字节
总结与思考
优点
- 功能完整:程序能够准确解析 BMP 文件并提取关键属性。
- 用户友好:通过命令行交互,用户可以轻松使用程序。
改进方向
- 支持更多格式:扩展程序以支持其他图像格式(如 JPEG、PNG)。
- 错误处理:增加更详细的错误提示和异常处理。
- 性能优化:优化文件读取和处理速度。
参考文献
- C语言从入门到项目实战
- BMP 文件格式解析
- C语言课程设计案例
附录代码
#include <stdio.h>#include <stdlib.h>#include <string.h>#define MAX_MENU 100 // 定义菜单项的最大数量#define MAX_ORDER 100 // 定义订单的最大数量// 定义菜单项结构体typedef struct {int id; // 菜品IDchar name[50]; // 菜品名称float price; // 菜品价格} MenuItem;// 定义订单结构体typedef struct {int order_id; // 订单IDchar customer_phone[20]; // 顾客电话char customer_name[50]; // 顾客姓名char address[100]; // 顾客地址char order_time[20]; // 订单时间MenuItem items[MAX_MENU]; // 订单包含的菜品列表int items_count; // 订单中菜品的数量float total_amount; // 订单总金额} Order;// 定义全局变量MenuItem menu[MAX_MENU] = {0};Order orders[MAX_ORDER] = {0};int menu_count = 0;int order_count = 0;// 函数声明void addMenuItem(); // 添加菜单项void modifyMenuItem(int id); // 修改菜单项void displayMenu(); // 显示菜单void placeOrder(); // 下订单void cancelOrder(int order_id); // 取消订单void searchOrderByID(int order_id); // 通过订单ID搜索订单void searchOrderByPhone(const char *phone); // 通过电话号码搜索订单void statistics(); // 统计信息void applyDiscount(float *amount); // 应用折扣void printOrder(const Order *order); // 打印订单详情void clearOrder(Order *order); // 清除订单数据// 主函数int main() {int choice;do {printf("\n1. 添加/修改菜单项\n2. 下订单\n3. 取消订单\n4. 搜索订单\n5. 统计信息\n6. 退出\n");printf("输入你的选择: ");scanf("%d", &choice);switch (choice) {case 1:addMenuItem();break;case 2:placeOrder();break;case 3:printf("输入要取消的订单ID: ");scanf("%d", &choice);cancelOrder(choice);break;case 4:printf("通过 (1) 订单ID 或 (2) 电话号码搜索: ");scanf("%d", &choice);if (choice == 1) {int order_id;printf("输入订单ID: ");scanf("%d", &order_id);searchOrderByID(order_id);} else if (choice == 2) {char phone[20];printf("输入电话号码: ");scanf("%s", phone);searchOrderByPhone(phone);}break;case 5:statistics();break;case 6:printf("退出系统.\n");break;default:printf("无效选择,请重新输入.\n");}} while (choice != 6);return 0;}// 添加菜单项void addMenuItem() {if (menu_count >= MAX_MENU) {printf("菜单已满,无法添加更多菜品。\n");return;}printf("输入菜品ID,名称和价格: ");scanf("%d %49s %f", &menu[menu_count].id, menu[menu_count].name, &menu[menu_count].price);menu_count++;}// 修改菜单项void modifyMenuItem(int id) {for (int i = 0; i < menu_count; i++) {if (menu[i].id == id) {printf("输入新的名称和价格: ");scanf("%49s %f", menu[i].name, &menu[i].price);return;}}printf("未找到菜品。\n");}// 显示菜单void displayMenu() {printf("菜单:\n");for (int i = 0; i < menu_count; i++) {printf("%d. %s - $%.2f\n", menu[i].id, menu[i].name, menu[i].price);}}// 下订单void placeOrder() {if (order_count >= MAX_ORDER) {printf("订单数量已达上限,无法下新订单。\n");return;}int item_id;float total = 0;orders[order_count].items_count = 0;displayMenu();printf("输入顾客的电话、姓名、地址和下单时间: ");scanf("%19s %49s %99s %19s", orders[order_count].customer_phone, orders[order_count].customer_name, orders[order_count].address, orders[order_count].order_time);while (1) {printf("输入菜品ID(0结束): ");scanf("%d", &item_id);if (item_id == 0) break;for (int i = 0; i < menu_count; i++) {if (menu[i].id == item_id) {if (orders[order_count].items_count < MAX_MENU) {orders[order_count].items[orders[order_count].items_count++] = menu[i];total += menu[i].price;} else {printf("一个订单中不能添加超过 %d 个菜品。\n", MAX_MENU);break;}}}}applyDiscount(&total);orders[order_count].total_amount = total;orders[order_count].order_id = order_count + 1; // 简单的订单ID生成逻辑printf("订单成功创建。订单ID: %d\n", orders[order_count].order_id);order_count++;}// 取消订单void cancelOrder(int order_id) {for (int i = 0; i < order_count; i++) {if (orders[i].order_id == order_id) {printf("订单 %d 已取消。\n", order_id);clearOrder(&orders[i]); // 清除订单数据for (int j = i; j < order_count - 1; j++) {memcpy(&orders[j], &orders[j + 1], sizeof(Order));}order_count--;return;}}printf("未找到订单。\n");}// 通过订单ID搜索订单void searchOrderByID(int order_id) {int found = 0; // 用于标记是否找到订单for (int i = 0; i < order_count; i++) {if (orders[i].order_id == order_id) {printOrder(&orders[i]);found = 1; // 标记找到订单break;}}if (!found) {printf("没有找到订单。\n");}}// 通过电话号码搜索订单void searchOrderByPhone(const char *phone) {int found = 0; // 用于标记是否找到订单for (int i = 0; i < order_count; i++) {if (strcmp(orders[i].customer_phone, phone) == 0) {printOrder(&orders[i]);found = 1; // 标记找到订单}}if (!found) {printf("没有找到该电话号码的订单。\n");}}// 统计信息void statistics() {// 示例统计信息 - 可以根据实际需求扩展int order_count_per_item[MAX_MENU] = {0};float total_revenue = 0;for (int i = 0; i < order_count; i++) {total_revenue += orders[i].total_amount;for (int j = 0; j < orders[i].items_count; j++) {int item_id = orders[i].items[j].id;order_count_per_item[item_id]++;}}printf("今日总收入: %.2f\n", total_revenue);for (int i = 0; i < menu_count; i++) {if (order_count_per_item[menu[i].id] > 0) {printf("%s 被订购了 %d 次。\n", menu[i].name, order_count_per_item[menu[i].id]);}}}// 应用折扣void applyDiscount(float *amount) {if (*amount > 300) *amount *= 0.85f;else if (*amount > 200) *amount *= 0.9f;else if (*amount > 100) *amount *= 0.95f;}// 打印订单详情void printOrder(const Order *order) {if (order == NULL) {printf("订单为空。\n");return;}// 打印订单头部信息printf("订单ID: %d\n", order->order_id);printf("顾客电话: %s\n", order->customer_phone);printf("顾客姓名: %s\n", order->customer_name);printf("地址: %s\n", order->address);printf("下单时间: %s\n", order->order_time);// 检查是否有订单项if (order->items_count == 0) {printf("该订单没有包含任何菜品。\n");} else {printf("订单项:\n");for (int i = 0; i < order->items_count; i++) {// 打印每个订单项的名称和价格printf(" - %s ($%.2f)\n", order->items[i].name, order->items[i].price);}}// 打印订单总金额printf("总金额: $%.2f\n", order->total_amount);}// 清除订单数据void clearOrder(Order *order) {memset(order, 0, sizeof(Order));}
嗨,我是LucianaiB。如果你觉得我的分享有价值,不妨通过以下方式表达你的支持:👍 点赞来表达你的喜爱,📁 关注以获取我的最新消息,💬 评论与我交流你的见解。我会继续努力,为你带来更多精彩和实用的内容。
点击这里👉LucianaiB ,获取最新动态,⚡️ 让信息传递更加迅速。

相关文章:
C语言之图像文件的属性
🌟 嗨,我是LucianaiB! 🌍 总有人间一两风,填我十万八千梦。 🚀 路漫漫其修远兮,吾将上下而求索。 图像文件属性提取系统设计与实现 目录 设计题目设计内容系统分析总体设计详细设计程序实现…...
LeetCode hot 100—分割等和子集
题目 给你一个 只包含正整数 的 非空 数组 nums 。请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。 示例 示例 1: 输入:nums [1,5,11,5] 输出:true 解释:数组可以分割成 [1, 5, 5] 和 [11] 。…...
高等数学同步测试卷 同济7版 试卷部分 上 做题记录 上册期中同步测试卷 B卷
上册期中同步测试卷 B卷 一、单项选择题(本大题共5小题,每小题3分,总计15分) 1. 2. 3. 4. 5. 由f(2/n), n→∞可知 2/n→0, 即x→0. 二、填空题(本大题共5小题,每小题3分,总计15分) 6. 7. 8. 9. 10. 三、求解下列各题(本大题共5小…...
【算法】快速排序、归并排序(非递归版)
目录 一、快速排序(非递归) 1.原理 2.实现 2.1 stack 2.2 partition(array,left,right) 2.3 pivot - 1 > left 二、归并排序(非递归) 1.原理 2.实现 2.1 gap 2.1.1 i 2*gap 2.1.2 gap * 2 2.1.3 gap < array.…...
python-将文本生成音频
将文本生成音频通常需要结合 文本转语音(TTS,Text-to-Speech) 工具或库来实现,比如 Google TTS (gtts)、Amazon Polly、Microsoft Azure TTS 等。 一、使用 Google TTS (gtts) 将文本生成音频 gtts 是一个简单易用的 Python 库&a…...
[王阳明代数讲义]语言模型核心代码调研
语言模型核心代码调研 基于Consciciteation的才气张量持续思考综述将文本生成建模为才气张量网络扩散过程,实现非自回归推理通过才气张量的群-拓扑流形交叉注意力实现多模态推理,将输入压缩到低维空间持续迭代提出「条件计算提前终止」机制,…...
4月19日记(补)算了和周日一块写了 4月20日日记
周六啊 昨天晚上又玩的太嗨了。睡觉的时候有点晚了,眼睛疼就没写日记。现在补上 实际上现在是20号晚上八点半了。理论上来说应该写今天的日记。 周六上午打比赛啦,和研究生,输了,我是替补没上场。没关系再练一练明天就可以变强…...
trivy开源安全漏洞扫描器——筑梦之路
开源地址:https://github.com/aquasecurity/trivy.git 可扫描的对象 容器镜像文件系统Git存储库(远程)虚拟机镜像Kubernetes 在容器镜像安全方面使用广泛,其他使用相对较少。 能够发现的问题 正在使用的操作系统包和软件依赖项…...
【实战中提升自己】内网安全部署之dot1x部署 本地与集成AD域的主流方式(附带MAC认证)
1 dot1x部署【用户名密码认证,也可以解决私接无线AP等功能】 说明:如果一个网络需要通过用户名认证才能访问内网,而认证失败只能访问外网与服务器,可以部署dot1x功能。它能实现的效果是,当内部用户输入正常的…...
[matlab]南海地形眩晕图代码
[matlab]南海地形眩晕图代码 请ChatGPT帮写个南海地形眩晕图代码 图片 图片 代码 .rtcContent { padding: 30px; } .lineNode {font-size: 12pt; font-family: "Times New Roman", Menlo, Monaco, Consolas, "Courier New", monospace; font-style: n…...
Web安全和渗透测试--day6--sql注入--part 1
场景: win11家庭版,edge浏览器 , sqlin靶场 定义: SQL 注入(SQL Injection)是一种常见的网络安全攻击方式,攻击者通过在 Web 应用程序中输入恶意的 SQL 代码,绕过应用程序的安全机…...
大模型在胆管结石(无胆管炎或胆囊炎)预测及治疗方案制定中的应用研究
目录 一、引言 1.1 研究背景与意义 1.2 研究目的 1.3 国内外研究现状 二、胆管结石相关理论基础 2.1 胆管结石概述 2.2 临床表现与诊断方法 2.3 传统治疗方法 三、大模型技术原理与应用优势 3.1 大模型基本原理 3.2 在医疗领域的应用潜力 3.3 用于胆管结石预测的可…...
MIT6.S081-lab4
MIT6.S081-lab4 注:本篇lab的前置知识在《MIT6.S081-lab3前置》 1. RISC-V assembly 第一个问题 Which registers contain arguments to functions? For example, which register holds 13 in main’s call to printf? 我们先来看看main干了什么: …...
精通 Spring Cache + Redis:避坑指南与最佳实践
Spring Cache 以其优雅的注解方式,极大地简化了 Java 应用中缓存逻辑的实现。结合高性能的内存数据库 Redis,我们可以轻松构建出响应迅速、扩展性强的应用程序。然而,在享受便捷的同时,一些常见的“坑”和被忽视的最佳实践可能会悄…...
[SpringBoot]快速入门搭建springboot
默认有spring基础,不会一行代码一行代码那么细致地讲。 SpringBoot的作用 Spring Boot是为了简化Spring应用的创建、运行、调试、部署等而出现的。就像我们整个SSM框架时,就常常会碰到版本导致包名对不上、Bean非法参数类型的一系列问题(原出…...
理解.NET Core中的配置Configuration
什么是配置 .NET中的配置,本质上就是key-value键值对,并且key和value都是字符串类型。 在.NET中提供了多种配置提供程序来对不同的配置进行读取、写入、重载等操作,这里我们以为.NET 的源码项目为例,来看下.NET中的配置主要是有…...
C++面试八股文:智能指针
一、了解哪些智能指针? 回答:智能指针是用于管理动态分配的内存,行为类似于指针,但又具有自动管理内存的能力,所以称为智能指针。 首先说一下 auto_ptr和unique_ptr,它们都是独占式指针,同一时…...
nohup命令使用说明
文章目录 如何在后台运行程序呢?如何正常运行代码重定向呢?nohup: ignoring input 如何在后台运行程序呢? 使用nohup命令即可, nohup python dataset/ReferESpatialDataset.py >>dataset_20250417.log 2>&1 &n…...
MYSQL “Too Many Connections“ 错误解决
1.查询当前连接数 show status like "Threads_connected"; 2.查询数据库最大连接数 show variables like "max_connections" 3.查询所有活动连接 show processlist; 4.根据查询结果观察是否有长时间未被释放的连接 参数解释 : 字段说明id连接的唯一…...
Linux `init 6` 相关命令的完整使用指南
Linux init 6 相关命令的完整使用指南—目录 一、init 系统简介二、init 6 的含义与作用三、不同 Init 系统下的 init 6 行为1. SysVinit(如 CentOS 6、Debian 7)2. systemd(如 CentOS 7、Ubuntu 16.04)3. Upstart(如 …...
【外研在线-注册/登录安全分析报告】
前言 由于网站注册入口容易被黑客攻击,存在如下安全问题: 暴力破解密码,造成用户信息泄露短信盗刷的安全问题,影响业务及导致用户投诉带来经济损失,尤其是后付费客户,风险巨大,造成亏损无底洞…...
【NLP 63、大模型应用 —— Agent】
人与人最大的差距就是勇气和执行力,也是唯一的差距 —— 25.4.16 一、Agent 相关工作 二、Agent 特点 核心特征: 1.专有场景(针对某个垂直领域) 2.保留记忆(以一个特定顺序做一些特定任务,记忆当前任务的前…...
React 打包
路由懒加载 原本的加载方式 #使用lazy()函数声明的路由页面 使用Suspense组件进行加载 使用CDN优化...
2025.4.14-2025.4.20学习周报
目录 摘要Abstract1. 文献阅读1.1 模型架构1.2 实验分析1.3 代码实践 总结 摘要 在本周阅读的论文中,作者提出了一种名为MGSFformer的空气质量预测模型。模型通过残差去冗余模块可以有效解耦多粒度数据间的信息重叠;时空注意力模块采用并行建模策略&…...
Spring 微服务解决了单体架构的哪些痛点?
1. 部署困难 (Deployment Difficulty & Risk) 单体痛点: 整体部署: 对单体应用的任何微小修改(哪怕只是一行代码),都需要重新构建、测试和部署整个庞大的应用程序。部署频率低: 由于部署过程复杂且风险高,发布周期通常很长&a…...
【1】云原生,kubernetes 与 Docker 的关系
Kubernetes?K8s? Kubernetes经常被写作K8s。其中的数字8替代了K和s中的8个字母——这一点倒是方便了发推,也方便了像我这样懒惰的人。 什么是云原生? 云原生: 它是一种构建和运行应用程序的方法,它包含&am…...
Kubernetes控制平面组件:APIServer 限流机制详解
云原生学习路线导航页(持续更新中) kubernetes学习系列快捷链接 Kubernetes架构原则和对象设计(一)Kubernetes架构原则和对象设计(二)Kubernetes架构原则和对象设计(三)Kubernetes控…...
springboot全局异常捕获处理
一、需求 实际项目中,经常抛出各种异常,不能直接抛出异常给前端,这样用户体验相当不好,用户看不懂你的Exception,对于一些sql异常,直接抛到页面上也不安全。所以有没有好的办法解决这些问题呢,当然有了&am…...
Flask(1): 在windows系统上部署项目1
1 前言 学习python也有段时间了,最近一个小项目要部署,正好把过程写下来。 在程序的结构上我选择了w/s模式,相比于c/s模式,无需考虑客户端的升级;框架我选择了flask,就是冲着轻量级去的,就是插件…...
【文献阅读】EndoNet A Deep Architecture for Recognition Tasks on Laparoscopic Videos
关于数据集的整理 Cholec80 胆囊切除手术视频数据集介绍 https://zhuanlan.zhihu.com/p/700024359 数据集信息 Cholec80 数据集 是一个针对内窥镜引导 下的胆囊切除手术视频流程识别数据集。数据集提供了每段视频中总共7种手术动作及总共7种手术工具的标注,标…...
