C语言之图像文件的属性


🌟 嗨,我是LucianaiB!

🌍 总有人间一两风,填我十万八千梦。

🚀 路漫漫其修远兮,吾将上下而求索。



图像文件属性提取系统设计与实现

目录

  1. 设计题目
  2. 设计内容
  3. 系统分析
  4. 总体设计
  5. 详细设计
  6. 程序实现
  7. 测试数据和运行结果
  8. 总结与思考
  9. 参考文献

设计题目

图像文件的属性提取


设计内容

题目描述

本项目的目标是编写一个 C 语言程序,能够读取 BMP 格式的图像文件,并提取图像的基本属性,如宽度、高度、颜色深度等。程序需要解析文件格式并提取属性,但不需要对图像进行渲染或处理。

题目要求

  1. 自动判断文件是否为 BMP 格式。
  2. 提取图像的灰度或彩色信息。
  3. 提取图像的宽度和高度(以像素为单位)。
  4. 计算图像所占的字节数。
  5. 将指定矩形区域内的像素值写入到文件。

输入/输出要求

  • 输入
    • 用户通过命令行输入图像文件路径。
    • 程序验证路径是否有效,文件是否存在。
  • 输出
    • 在控制台输出图像属性信息。
    • 若输入无效,输出错误提示信息。

系统分析

本项目旨在实现一个图像文件属性提取工具,能够快速解析 BMP 文件格式并提取关键信息。系统需要具备以下功能:

  1. 文件格式验证。
  2. 属性提取(宽度、高度、颜色深度等)。
  3. 数据持久化(将像素值写入文件)。
  4. 用户友好的交互界面。

总体设计

系统采用模块化设计,主要分为以下几个模块:

  1. 文件解析模块:负责读取 BMP 文件并验证格式。
  2. 属性提取模块:提取图像的基本属性。
  3. 数据处理模块:处理像素数据并写入文件。
  4. 用户界面模块:提供命令行交互界面。

详细设计

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 函数功能描述

  1. 读取 BMP 文件

    int readBMP(const char* filename, BMPFileHeader* fileHeader, BMPInfoHeader* infoHeader);
    

    功能:读取 BMP 文件并验证格式。

  2. 提取图像属性

    void extractAttributes(const BMPInfoHeader* infoHeader);
    

    功能:提取图像的宽度、高度、颜色深度等属性。

  3. 写入像素数据

    void writePixelData(const char* outputFilename, const unsigned char* pixelData, int dataSize);
    

    功能:将指定区域的像素值写入文件。

  4. 主函数

    int main(int argc, char* argv[]);
    

    功能:处理用户输入,调用文件解析和属性提取模块。

3.3 主要函数流程图

有效
无效
BMP
非BMP
开始
读取文件路径
验证路径
读取BMP文件
输出错误信息
验证文件格式
提取属性
输出属性信息
写入像素数据
结束

程序实现

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字节

总结与思考

优点

  1. 功能完整:程序能够准确解析 BMP 文件并提取关键属性。
  2. 用户友好:通过命令行交互,用户可以轻松使用程序。

改进方向

  1. 支持更多格式:扩展程序以支持其他图像格式(如 JPEG、PNG)。
  2. 错误处理:增加更详细的错误提示和异常处理。
  3. 性能优化:优化文件读取和处理速度。

参考文献

  1. C语言从入门到项目实战
  2. BMP 文件格式解析
  3. 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 ,获取最新动态,⚡️ 让信息传递更加迅速。

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

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

相关文章

opencv--基础

opencv OpenCV是一个实现数字图像处理和计算机视觉通用算法的开源跨平台库。 链接 opencv中的cv是什么意思 在OpenCV中&#xff0c;"cv" 是 "Computer Vision"&#xff08;计算机视觉&#xff09; 的缩写。 opencv的实现语言 opencv的底层实现代码是使…

Java创建对象的方式

1、通过new关键字创建新对象 用new关键字创建对象是我们在开发中最常用的方式&#xff0c;new关键字会为我们在堆内存中开辟一块空间以存放对象的引用&#xff08;包含对象本身以及内部属性的引用&#xff09;。 2、通过newInstance()方法创建新对象 newInstance()方法本质上是…

构建具备推理与反思能力的高级 Prompt:LLM 智能代理设计指南

在构建强大的 AI 系统&#xff0c;尤其是基于大语言模型&#xff08;LLM&#xff09;的智能代理&#xff08;Agent&#xff09;时&#xff0c;Prompt 设计的质量决定了系统的智能程度。传统 Prompt 通常是简单的问答或填空式指令&#xff0c;而高级任务需要更具结构性、策略性和…

猪行为视频数据集

猪行为数据集包含 23 天(超过 6 周)的日间猪行为视频,这些视频由近乎架空的摄像机拍摄。视频已配准颜色和深度信息。数据以每秒 6 帧的速度捕获,并以 1800 帧(5 分钟)为一批次进行存储。大多数帧显示 8 头猪。 这里可以看到颜色和深度图像的示例: 喂食器位于图片底部中…

C++运算符重载详解

C++ 中的运算符重载允许为用户自定义类型(类或结构体)赋予运算符特定功能,使其操作更直观。以下是运算符重载的关键点: 1. 基本语法 成员函数重载:运算符作为类的成员函数,左操作数为当前对象 (this),右操作数为参数。 class Complex {public:Complex operator+(const …

deep-share开源浏览器扩展,用于分享 DeepSeek 对话,使用户能够将对话内容保存为图片或文本以便轻松分享

一、软件介绍 文末提供程序和源码下载学习 deep-share开源浏览器扩展&#xff0c;用于分享 DeepSeek 对话&#xff0c;使用户能够将对话内容保存为图片或文本以便轻松分享。 二、软件功能 One-click capture of DeepSeek chat content一键捕获 DeepSeek 聊天内容Support sha…

Unity之如何实现RenderStreaming视频推流

文章目录 前言引入 UnityRenderStreaming 的好处教程步骤 1:设置环境步骤 2: 创建项目步骤 3:安装软件包步骤 5:下载示例步骤 6:检查配置环境步骤 7:打开推流场景步骤 8: 准备用于流式传输的WebServer应用程序步骤 9: 运行 示例场景步骤 10:检查视频是否在浏览器中显示…

30天开发操作系统 第26天 -- 为窗口移动提速

前言 昨天我们增加了可同时启动的应用程序的数量&#xff0c;窗口也跟着变多了&#xff0c;整个画面变得热闹起来。 话说&#xff0c;在对比color.hrb和color2.hrb的时候我们需要移动窗口&#xff0c;那个时候笔者感到窗口移动的速度很慢。在真机环境下的速度还算可以接受&…

9.QT-显示类控件|Label|显示不同格式的文本|显示图片|文本对齐|自动换行|缩进|边距|设置伙伴(C++)

Label QLabel 可以⽤来显⽰⽂本和图⽚ 属性说明textQLabel中的⽂本textFormat⽂本的格式.• Qt::PlainText 纯⽂本• Qt::RichText 富⽂本(⽀持html标签)• Qt::MarkdownText markdown格式• Qt::AutoText 根据⽂本内容⾃动决定⽂本格式pixmapQLabel 内部包含的图⽚.scaledCo…

非参数检验题目集

非参数检验题目集 对医学计量资料成组比较&#xff0c;相对参数检验来说&#xff0c;非参数秩和检验的优点是&#xff08; &#xff09; A. 适用范围广 B. 检验效能高 C. 检验结果更准确 D. 充分利用资料信息 E. 不易出现假阴性错误 对于计量资料的比较&#xff0c;在满足参数…

libdxfrw库使用总结

在 Win11VS2022CMake 平台编译 libdxfrw 库的挑战与应对 在当今数字化设计与开发领域&#xff0c;高效处理 CAD 文件格式如 DXF 是众多项目的关键需求。libdxfrw 库作为一种功能强大的工具&#xff0c;能助力开发者精准解析与写入 DXF 文件&#xff0c;使其在众多应用场景中备…

C++学习:六个月从基础到就业——内存管理:RAII原则

C学习&#xff1a;六个月从基础到就业——内存管理&#xff1a;RAII原则 本文是我C学习之旅系列的第十九篇技术文章&#xff0c;也是第二阶段"C进阶特性"的第四篇&#xff0c;主要介绍C中的RAII原则及其在资源管理中的应用。查看完整系列目录了解更多内容。 引言 在…

【愚公系列】《Python网络爬虫从入门到精通》056-Scrapy_Redis分布式爬虫(Scrapy-Redis 模块)

&#x1f31f;【技术大咖愚公搬代码&#xff1a;全栈专家的成长之路&#xff0c;你关注的宝藏博主在这里&#xff01;】&#x1f31f; &#x1f4e3;开发者圈持续输出高质量干货的"愚公精神"践行者——全网百万开发者都在追更的顶级技术博主&#xff01; &#x1f…

PyTorch基础笔记

PyTorch张量 多维数组&#xff1a;张量可以是标量&#xff08;0D&#xff09;、向量&#xff08;1D&#xff09;、矩阵&#xff08;2D&#xff09;或更高维的数据&#xff08;3D&#xff09;。 数据类型&#xff1a;支持多种数据类型&#xff08;如 float32, int64, bool 等&a…

OSCP - Proving Grounds - Sar

主要知识点 路径爆破cronjob 脚本劫持提权 具体步骤 依旧nmap 开始,开放了22和80端口 Nmap scan report for 192.168.192.35 Host is up (0.43s latency). Not shown: 65524 closed tcp ports (reset) PORT STATE SERVICE VERSION 22/tcp open ssh Open…

存储/服务器内存的基本概念简介

为什么写这个文章&#xff1f;今天处理一个powerstore 3000T 控制器&#xff0c;控制器上电后&#xff0c;亮一下灯就很快熄灭了&#xff0c;然后embedded module上和io module不加电&#xff0c;过一整子系统自动就下电了&#xff0c;串口没有任何输出。刚开始判断是主板的问题…

软件开发指南——GUI 开发方案推荐

1. LVGL (Light and Versatile Graphics Library) 适用场景&#xff1a;嵌入式设备、资源受限环境 优势&#xff1a; 专为嵌入式设计的开源 GUI 库&#xff0c;内存占用极小&#xff08;最低仅需 64KB RAM&#xff09;支持触摸屏、硬件加速&#xff08;如 STM32 的 LTDC&…

8 编程笔记全攻略:Markdown 语法精讲、Typora 编辑器全指南(含安装激活、基础配置、快捷键详解、使用技巧)

1 妙笔在手&#xff0c;编程无忧&#xff01; 1.1 编程为啥要做笔记&#xff1f;这答案绝了&#xff01; 嘿&#xff0c;各位键盘魔法师&#xff01;学编程不记笔记&#xff0c;就像吃火锅不配冰可乐 —— 爽到一半直接噎住&#xff01;你以为自己脑子是顶配 SSD&#xff0c;结…

LeetCode -- Flora -- edit 2025-04-16

1.两数之和 1. 两数之和 给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个 整数&#xff0c;并返回它们的数组下标。 你可以假设每种输入只会对应一个答案&#xff0c;并且你不能使用两次相同的元素。 你可以按…

web后端语言下篇

#作者&#xff1a;允砸儿 #日期&#xff1a;乙巳青蛇年 三月廿一 笔者今天将web后端语言PHP完结一下&#xff0c;后面还会写一个关于python的番外。 PHP函数 PHP函数它和笔者前面写的js函数有些许类似&#xff0c;都是封装的概念。将实现某一功能的代码块封装到一个结构中…