C语言实例_获取文件MD5值

一、MD5介绍

MD5(Message Digest Algorithm 5)是一种常用的哈希函数算法。将任意长度的数据作为输入,并生成一个唯一的、固定长度(通常是128位)的哈希值,称为MD5值。MD5算法以其高度可靠性和广泛应用而闻名。

image-20230626224839130

MD5算法主要具备以下特点:

(1)不可逆性:给定MD5值无法通过逆运算得到原始数据。

(2)唯一性:不同的输入数据会生成不同的MD5值。

(3)高效性:对于给定的数据,计算其MD5值是非常快速的。

MD5值的应用场景包括:

(1)数据完整性验证:MD5值可以用于验证文件是否在传输过程中被篡改。发送方计算文件的MD5值并发送给接收方,接收方在接收到文件后重新计算MD5值,然后与发送方的MD5值进行比较,如果一致,则说明文件未被篡改。

(2)密码存储:在许多系统中,用户密码通常不会以明文形式存储,而是将其转换为MD5值后存储。当用户登录时,系统会将用户输入的密码转换为MD5值,然后与存储的MD5值进行比较,以验证密码的正确性。

(3)安全认证:MD5值也可用于数字证书等安全认证中,用于验证文件的完整性和认证信息的真实性。

(4)数据指纹:MD5值可以作为数据的唯一标识符,用于快速比对和查找重复数据。

二、示例代码

2.1 获取数据MD5值(openssl库)

在C语言中获取一段数据的MD5值,可以使用现有的第三方库实现。以下是一个使用 OpenSSL 库计算数据的MD5值的示例代码:

(1)需要安装 OpenSSL 库(如果尚未安装)并包含相关头文件:

#include <stdio.h>
#include <stdlib.h>
#include <openssl/md5.h>

(2)创建一个子函数来计算数据的MD5值:

void calculate_md5(const unsigned char* data, size_t length, unsigned char* md5_hash) {MD5_CTX ctx;MD5_Init(&ctx);MD5_Update(&ctx, data, length);MD5_Final(md5_hash, &ctx);
}

该函数接受三个参数:data 为待计算的数据指针,length 为数据长度,md5_hash 为存储MD5值的数组。

下面是一个完整的程序,展示如何调用以上子函数并打印MD5值:

#include <stdio.h>
#include <stdlib.h>
#include <openssl/md5.h>void calculate_md5(const unsigned char* data, size_t length, unsigned char* md5_hash) {MD5_CTX ctx;MD5_Init(&ctx);MD5_Update(&ctx, data, length);MD5_Final(md5_hash, &ctx);
}void print_md5(const unsigned char* md5_hash) {for (int i = 0; i < MD5_DIGEST_LENGTH; i++) {printf("%02x", md5_hash[i]);}printf("\n");
}int main() {const unsigned char data[] = "Hello, World!";size_t length = sizeof(data) - 1; // 减去字符串末尾的空字符unsigned char md5_hash[MD5_DIGEST_LENGTH];calculate_md5(data, length, md5_hash);printf("MD5: ");print_md5(md5_hash);return 0;
}

这个示例程序将输出一段数据的MD5值。可以将待计算的数据存储在 data 数组中,并根据需要调整数据长度。

这里使用的是 OpenSSL 提供的 MD5 函数。在编译时,需要链接 OpenSSL 库。在 Linux 系统上,可以使用 -lssl -lcrypto 参数进行链接。在 Windows 系统上,需要下载并安装 OpenSSL 库,并配置正确的链接路径和库文件名称。

2.2 获取文件的MD5值(openssl库)

以下是使用 OpenSSL 库计算文件的MD5值的示例代码:

(1)需要安装 OpenSSL 库(如果尚未安装)并包含相关头文件:

#include <stdio.h>
#include <stdlib.h>
#include <openssl/md5.h>

(2)创建一个子函数来计算文件的MD5值:

void calculate_file_md5(const char* filename, unsigned char* md5_hash) {FILE* file = fopen(filename, "rb");if (file == NULL) {printf("Failed to open file: %s\n", filename);return;}MD5_CTX ctx;MD5_Init(&ctx);unsigned char buffer[1024];size_t read;while ((read = fread(buffer, 1, sizeof(buffer), file)) != 0) {MD5_Update(&ctx, buffer, read);}fclose(file);MD5_Final(md5_hash, &ctx);
}

该函数接受两个参数:filename 为待计算的文件名,md5_hash 为存储MD5值的数组。

下面是一个完整的示例程序,展示如何调用以上子函数并打印文件的MD5值:

#include <stdio.h>
#include <stdlib.h>
#include <openssl/md5.h>void calculate_file_md5(const char* filename, unsigned char* md5_hash) {// ... 函数实现见上文 ...void print_md5(const unsigned char* md5_hash) {for (int i = 0; i < MD5_DIGEST_LENGTH; i++) {printf("%02x", md5_hash[i]);}printf("\n");
}int main() {const char* filename = "path/to/file";unsigned char md5_hash[MD5_DIGEST_LENGTH];calculate_file_md5(filename, md5_hash);printf("MD5: ");print_md5(md5_hash);return 0;
}

这个示例程序将打开指定文件并计算其MD5值。需要将文件路径存储在 filename 字符串中,并根据需要调整该字符串。

请这里使用的是 OpenSSL 提供的 MD5 函数。在编译时,需要链接 OpenSSL 库。在 Linux 系统上,可以使用 -lssl -lcrypto 参数进行链接。在 Windows 系统上,需要下载并安装 OpenSSL 库,并配置正确的链接路径和库文件名称。

2.3 自己写算法获取MD5值

实现MD5算法比较复杂,涉及位操作、逻辑运算、位移等。

以下是一个简化版本的纯C语言MD5算法实现:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>typedef unsigned char uint8;
typedef unsigned int uint32;// MD5常量定义
const uint32 MD5_CONSTANTS[] = {0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee,0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501,0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be,0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821,0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa,0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8,0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed,0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a,0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c,0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70,0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05,0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665,0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039,0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1,0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1,0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391
};// 循环左移
#define LEFT_ROTATE(x, n) (((x) << (n)) | ((x) >> (32 - (n))))// 转换为大端字节序
void to_big_endian(uint32 value, uint8* buffer) {buffer[0] = (uint8)(value & 0xff);buffer[1] = (uint8)((value >> 8) & 0xff);buffer[2] = (uint8)((value >> 16) & 0xff);buffer[3] = (uint8)((value >> 24) & 0xff);
}// 处理消息块
void process_block(const uint8* block, uint32* state) {uint32 a = state[0];uint32 b = state[1];uint32 c = state[2];uint32 d = state[3];uint32 m[16];// 将消息块划分为16个32位字,并进行字节序转换for (int i = 0; i < 16; i++) {m[i] = (((uint32)block[i * 4 + 0]) << 0) |(((uint32)block[i * 4 + 1]) << 8) |(((uint32)block[i * 4 + 2]) << 16) |(((uint32)block[i * 4 + 3]) << 24);}// MD5循环运算for (int i = 0; i < 64; i++) {uint32 f, g;if (i < 16) {f = (b & c) | ((~b) & d);g = i;} else if (i < 32) {f = (d & b) | ((~d) & c);g = (5 * i + 1) % 16;} else if (i < 48) {f = b ^ c ^ d;g = (3 * i + 5) % 16;} else {f = c ^ (b | (~d));g = (7 * i) % 16;}uint32 temp = d;d = c;c = b;b = b + LEFT_ROTATE((a + f + MD5_CONSTANTS[i] + m[g]), 7);a = temp;}// 更新状态state[0] += a;state[1] += b;state[2] += c;state[3] += d;
}// 计算MD5值
void calculate_md5(const uint8* message, size_t length, uint8* digest) {// 初始化状态uint32 state[4] = { 0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476 };// 填充消息size_t padded_length = ((length + 8) / 64 + 1) * 64;uint8* padded_message = (uint8*)calloc(padded_length, 1);memcpy(padded_message, message, length);padded_message[length] = 0x80;  // 添加一个1to_big_endian((uint32)(length * 8), padded_message + padded_length - 8);  // 添加长度(以位为单位)// 处理消息块for (size_t i = 0; i < padded_length; i += 64) {process_block(padded_message + i, state);}// 生成摘要for (int i = 0; i < 4; i++) {to_big_endian(state[i], digest + i * 4);}free(padded_message);
}// 打印MD5值
void print_md5(const uint8* digest) {for (int i = 0; i < 16; i++) {printf("%02x", digest[i]);}printf("\n");
}int main() {const char* message = "Hello, World!";size_t length = strlen(message);uint8 digest[16];calculate_md5((const uint8*)message, length, digest);printf("MD5: ");print_md5(digest);return 0;
}

这个程序可以计算给定字符串的MD5值。将待计算的数据存储在 message 字符串中,根据需要调整数据长度。

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

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

相关文章

全球磁强计市场价值约为16.2亿美元,预测期内将以超过5.21%的增长率增长

磁强计是一种用于测量磁场强度和方向的仪器。它可以检测和测量地球磁场、物体的磁性、地下矿藏、磁性材料等。磁强计在地球科学、物理学、地质学、勘探、矿业等领域具有广泛的应用。 根据阿谱尔&#xff08;APO&#xff09;的统计及预测&#xff0c;2022年全球磁强计市场价值约…

ts与vue

ts与Vue 如果你已经学习了typeScript,但不知道如何在vue项目中使用&#xff0c;那么这篇文章将会很适合你。参考千峰教育 kerwin视频 1.会自动推导&#xff0c;隐士推导。提示 类型系统。 独立模块。 isolatedModules选项&#xff1a;是否配置为独立的模块。 减少报错 let …

dispatcherServlet在tomcat启动时如何被加载(1)

目录 在springboot工程中, 如何添加一个servlet呢? 方法1 : 使用WebServlet注解 方法2 : 使用ServletRegistrationBean进行注册 springmvc 采用的就是方式2和springboot集成的, 看一下源码 springboot 字段装配里面有这个类, 看一下源码 90行, 创建了一个DispatcherServlet对象…

解决Adobe Flash Player已被屏蔽

问题&#xff1a;该插件不支持 原因&#xff1a;现在浏览器默认禁用flash 博主当前使用的是谷歌浏览器Chrome 2个主要方法都已经失效 搜索一圈后&#xff0c;之前博客给出的2个主要方法都已经失效。 1、flash.cn 下载本地播放器 2、在chrome中打开flash的禁用开关 2023年解…

LangChain源码逐行解密之系统(二)

LangChain源码逐行解密之系统 20.2 serapi.py源码逐行剖析 我们可以看一下Google查询的例子,在LangChain中有多种实现的方式。 如图20-5所示,在utilities的serpapi.py代码文件中实现了SerpAPIWrapper。 图20- 5 utilities的serpapi.py的SerpAPIWrapper 在langchain目录的se…

如何避免爬虫IP被屏蔽

各位爬友们好&#xff0c;作为一名专业的爬虫代理提供者&#xff0c;我要和大家分享一些避免爬虫IP被屏蔽的实用技巧。你知道吗&#xff0c;当我们爬取数据的时候&#xff0c;很容易被目标网站识别出来并封禁我们的IP地址&#xff0c;导致无法继续爬取数据。这个问题困扰了很多…

7. 实现 API 自动生成

目录 1. pom.xml中引用依赖 2. 引入相关的依赖 3. 编写配置类 4. application.yml 中添加配置 5. API 常用注解 6. 访问 API 列表 7. API 导入 Postman 使用 Springfox Swagger生成 API&#xff0c;并导入 Postman&#xff0c;完成API单元测试。 Swagger 简介&#xff1a;Swag…

执行Lua脚本后一直查询不到Redis中的数据(附带问题详细排查过程,一波三折)

文章目录 执行Lua脚本后一直查询不到Redis中的数据&#xff08;附带详细问题排查过程&#xff0c;一波三折&#xff09;问题背景问题1&#xff1a;Lua脚本无法切库问题2&#xff1a;RedisTemlate切库报错问题3&#xff1a;序列化导致数据不一致问题4&#xff1a;Lua脚本中单引号…

Flask Web开发实战(狼书)| 笔记第1、2章

前言 2023-8-11 以前对网站开发萌生了想法&#xff0c;又有些急于求成&#xff0c;在B站照着视频敲了一个基于flask的博客系统。但对于程序的代码难免有些囫囵吞枣&#xff0c;存在许多模糊或不太理解的地方&#xff0c;只会照葫芦画瓢。 而当自己想开发一个什么网站的时&…

ubuntu部署haproxy

HAProxy是可提供高可用性、负载均衡以及基于TCP和HTTP应用的代理. 1、更新系统报 通过在终端中运行以下命令&#xff0c;确保所有系统包都是最新的 sudo apt updatesudo apt upgrade2、安装Haproxy sudo apt install haproxy设置开机自动启动haproxy服务 sudo systemctl en…

Lnton羚通关于如何解决nanoPC-T4 upgrade报错问题?

nanoPC-T4 在 ​​# sudo apt update 和 sudo apt upgrade​​升级或安装软件 ​​sudo apt install xxx​​时遇到以下问题&#xff1a;​​Failed to set up interface with /etc/hostapd/​ Setting up hostapd (2:2.6-15ubuntu2.8) ... Job for hostapd.service failed be…

ssm+vue医院住院管理系统源码和论文PPT

ssmvue医院住院管理系统源码和论文PPT012 开发工具&#xff1a;idea 数据库mysql5.7(mysql5.7最佳) 数据库链接工具&#xff1a;navcat,小海豚等 开发技术&#xff1a;java ssm tomcat8.5 摘 要 随着时代的发展&#xff0c;医疗设备愈来愈完善&#xff0c;医院也变成人们生…

基于IMX6ULLmini的linux裸机开发系列一:汇编点亮LED

思来想去还是决定记录一下点灯&#xff0c;毕竟万物皆点灯嘛 编程步骤 使能GPIO时钟 设置引脚复用为GPIO 设置引脚属性(上下拉、速率、驱动能力) 控制GPIO引脚输出高低电平 使能GPIO时钟 其实和32差不多 先找到控制LED灯的引脚&#xff0c;也就是原理图 文件名 C:/Us…

AUTOSAR NvM Block的三种类型

Native NVRAM block Native block是最基础的NvM Block&#xff0c;可以用来存储一个数据&#xff0c;可以配置长度、CRC等。 Redundant NVRAM block Redundant block就是在Native block的基础上再加一个冗余块&#xff0c;当Native block失效&#xff08;读取失败或CRC校验失…

剑指offer44.数字序列中某一位的数字

最后一道题&#xff0c;我一定要自己做出来&#xff0c;想了不到一个小时想法差不多成熟了&#xff0c;但是有一个小细节出问题了&#xff0c;这个问题我在idea上debug都没debug出来。我先讲我的题解然后再讲我这个小问题出在哪里吧。以下是我的代码&#xff1a; class Soluti…

PHP手术麻醉系统源码,自动生成麻醉和护理医疗文书

一套手术麻醉系统源码&#xff0c;可二次开发 手术室麻醉临床信息系统&#xff08;AIMS&#xff09;是应用于医院手术室、麻醉科室的计算机软件系统。该系统针对整个围术期&#xff0c;对病人进行全程跟踪与信息管理&#xff0c;自动集成病人HIS、LIS、RIS、PACS信息&#xff0…

【Spring源码】小白速通解析Spring源码,从0到1,持续更新!

Spring源码 参考资料 https://www.bilibili.com/video/BV1Tz4y1a7FM https://www.bilibili.com/video/BV1iz4y1b75q bean工厂 DefaultListableBeanFactory&#xff08;最原始&#xff09; bean的生命周期 创建&#xff08;实例化&#xff09;–>依赖注入–>-初始化…

数据生成 | MATLAB实现WGAN生成对抗网络数据生成

数据生成 | MATLAB实现WGAN生成对抗网络数据生成 目录 数据生成 | MATLAB实现WGAN生成对抗网络数据生成生成效果基本描述程序设计参考资料 生成效果 基本描述 1.WGAN生成对抗网络&#xff0c;数据生成&#xff0c;样本生成程序&#xff0c;MATLAB程序&#xff1b; 2.适用于MATL…

【游戏评测】河洛群侠传一周目玩后感

总游戏时长接近100小时&#xff0c;刚好一个月。 这两天费了点劲做了些成就&#xff0c;刷了等级&#xff0c;把最终决战做了。 总体感觉还是不错的。游戏是开放世界3D游戏&#xff0c;Unity引擎&#xff0c;瑕疵很多&#xff0c;但胜在剧情扎实&#xff0c;天赋系统、秘籍功法…

JDBC Vertica Source Connector 使用文档

支持以下引擎 Spark Flink SeaTunnel Zeta 关键特性 批处理 精确一次性处理 列投影 并行处理 支持用户自定义拆分 支持查询 SQL 并实现投影效果 描述 通过 JDBC 读取外部数据源数据。 支持的数据源信息 DatasourceSupported versionsDriverUrlMavenVerticaDifferent depen…