当前位置: 首页 > article >正文

从零手搓SM3国密算法:用C++一步步实现哈希函数(附完整可运行代码)

从零手搓SM3国密算法用C一步步实现哈希函数附完整可运行代码密码学算法的魅力在于它用数学的确定性构建了数字世界的安全基石。当我们谈论哈希函数时开发者往往满足于调用现成的库函数却错过了理解算法精妙之处的机会。本文将带你从零开始用C实现SM3国密哈希算法这个过程不仅是代码编写更是一次密码学思维的深度训练。1. 环境准备与基础工具函数1.1 开发环境配置实现SM3算法需要基本的C开发环境支持C11及以上标准的编译器GCC/Clang/MSVC均可文本编辑器或IDEVS Code/CLion等调试工具gdb/lldb或IDE内置调试器建议在Linux/macOS下开发可以更方便地使用命令行工具进行编译和测试。创建项目目录结构sm3_implement/ ├── include/ │ └── sm3.h # 算法声明 ├── src/ │ ├── sm3.cpp # 算法实现 │ └── main.cpp # 测试代码 └── Makefile # 构建配置1.2 基础数据类型转换SM3算法处理的是二进制位操作我们需要实现各种进制间的转换函数// 十六进制字符到4位二进制的映射 const std::unordered_mapchar, std::string hexToBinMap { {0, 0000}, {1, 0001}, {2, 0010}, {3, 0011}, {4, 0100}, {5, 0101}, {6, 0110}, {7, 0111}, {8, 1000}, {9, 1001}, {a, 1010}, {b, 1011}, {c, 1100}, {d, 1101}, {e, 1110}, {f, 1111} }; std::string hex_to_binary(const std::string hex) { std::string binary; for (char c : hex) { auto it hexToBinMap.find(tolower(c)); if (it ! hexToBinMap.end()) { binary it-second; } } return binary; }注意在实际实现中位操作比字符串操作效率更高。这里使用字符串是为了代码可读性生产环境应考虑优化。2. SM3算法核心组件实现2.1 消息填充机制SM3的消息填充遵循以下规则在消息末尾添加一个1位添加若干个0位直到长度满足 (长度 % 512) 448最后64位表示原始消息的位长度std::string sm3_padding(const std::string message) { // 转换为二进制字符串 std::string binary; for (char c : message) { binary std::bitset8(c).to_string(); } uint64_t original_length binary.size(); binary 1; // 添加终止位 // 填充0直到长度 ≡ 448 mod 512 while (binary.size() % 512 ! 448) { binary 0; } // 添加原始长度(64位大端序) binary std::bitset64(original_length).to_string(); return binary; }2.2 消息扩展函数消息扩展将512位的消息分组扩展为132个字W0-W67和W0-W63void message_expansion(const std::string block, std::arrayuint32_t, 68 W, std::arrayuint32_t, 64 W_prime) { // 初始化前16个字 for (int i 0; i 16; i) { W[i] std::stoul(block.substr(i * 32, 32), nullptr, 2); } // 计算W16-W67 for (int j 16; j 68; j) { uint32_t temp W[j-16] ^ W[j-9] ^ (rotate_left(W[j-3], 15)); W[j] P1(temp) ^ (rotate_left(W[j-13], 7)) ^ W[j-6]; } // 计算W0-W63 for (int j 0; j 64; j) { W_prime[j] W[j] ^ W[j4]; } }其中rotate_left和P1函数实现如下constexpr uint32_t rotate_left(uint32_t x, uint32_t n) { return (x n) | (x (32 - n)); } constexpr uint32_t P0(uint32_t x) { return x ^ rotate_left(x, 9) ^ rotate_left(x, 17); } constexpr uint32_t P1(uint32_t x) { return x ^ rotate_left(x, 15) ^ rotate_left(x, 23); }3. 压缩函数与迭代处理3.1 压缩函数实现压缩函数是SM3的核心处理扩展后的消息和当前哈希值void compression_function(const std::arrayuint32_t, 68 W, const std::arrayuint32_t, 64 W_prime, std::arrayuint32_t, 8 V) { uint32_t A V[0], B V[1], C V[2], D V[3]; uint32_t E V[4], F V[5], G V[6], H V[7]; for (int j 0; j 64; j) { uint32_t SS1 rotate_left(rotate_left(A, 12) E rotate_left(T(j), j), 7); uint32_t SS2 SS1 ^ rotate_left(A, 12); uint32_t TT1 FF(A, B, C, j) D SS2 W_prime[j]; uint32_t TT2 GG(E, F, G, j) H SS1 W[j]; D C; C rotate_left(B, 9); B A; A TT1; H G; G rotate_left(F, 19); F E; E P0(TT2); } V[0] ^ A; V[1] ^ B; V[2] ^ C; V[3] ^ D; V[4] ^ E; V[5] ^ F; V[6] ^ G; V[7] ^ H; }辅助函数实现constexpr uint32_t T(int j) { return (j 16) ? 0x79CC4519 : 0x7A879D8A; } constexpr uint32_t FF(uint32_t X, uint32_t Y, uint32_t Z, int j) { return (j 16) ? (X ^ Y ^ Z) : ((X Y) | (X Z) | (Y Z)); } constexpr uint32_t GG(uint32_t X, uint32_t Y, uint32_t Z, int j) { return (j 16) ? (X ^ Y ^ Z) : ((X Y) | ((~X) Z)); }3.2 迭代压缩过程完整的SM3算法处理流程std::string sm3_hash(const std::string message) { // 初始值IV std::arrayuint32_t, 8 V { 0x7380166F, 0x4914B2B9, 0x172442D7, 0xDA8A0600, 0xA96F30BC, 0x163138AA, 0xE38DEE4D, 0xB0FB0E4E }; // 消息填充 std::string padded sm3_padding(message); size_t blocks padded.size() / 512; // 处理每个512位块 for (size_t i 0; i blocks; i) { std::string block padded.substr(i * 512, 512); std::arrayuint32_t, 68 W {0}; std::arrayuint32_t, 64 W_prime {0}; message_expansion(block, W, W_prime); compression_function(W, W_prime, V); } // 生成最终哈希值 std::stringstream ss; for (uint32_t word : V) { ss std::hex std::setw(8) std::setfill(0) word; } return ss.str(); }4. 测试验证与性能优化4.1 标准测试向量验证使用官方测试用例验证实现正确性void test_sm3() { struct TestCase { std::string input; std::string expected; }; std::vectorTestCase tests { {abc, 66c7f0f462eeedd9d1f2d46bdc10e4e24167c4875cf2f7a2297da02b8f4ba8e0}, {abcdabcdabcdabcdabcdabcdabcdabcdabcdabcdabcdabcdabcdabcdabcdabcd, debe9ff92275b8a138604889c18e5a4d6fdb70e5387e5765293dcba39c0c5732} }; for (const auto test : tests) { std::string result sm3_hash(test.input); std::cout Input: test.input \n; std::cout Expected: test.expected \n; std::cout Actual: result \n; std::cout (result test.expected ? PASS : FAIL) \n\n; } }4.2 性能优化技巧原始实现可以通过以下方式优化减少内存分配预分配缓冲区避免频繁的字符串操作使用位操作替代算术运算如模加运算可以用掩码实现循环展开手动展开关键循环减少分支预测失败SIMD指令利用现代CPU的并行计算能力优化后的模加运算实现inline uint32_t mod_add(uint32_t a, uint32_t b) { uint32_t sum a b; return sum; // 在32位系统中自然溢出等同于模2^32 }4.3 错误处理与边界条件健壮的实现需要考虑各种边界情况std::string sm3_hash(const std::string message) { if (message.empty()) { // 处理空输入的特殊情况 return 1ab21d8355cfa17f8e61194831e81a8f22bec8c728fefb747ed035eb5082aa2b; } try { // ...正常处理流程... } catch (const std::exception e) { std::cerr SM3 hash error: e.what() std::endl; return ; } }

相关文章:

从零手搓SM3国密算法:用C++一步步实现哈希函数(附完整可运行代码)

从零手搓SM3国密算法:用C一步步实现哈希函数(附完整可运行代码) 密码学算法的魅力在于,它用数学的确定性构建了数字世界的安全基石。当我们谈论哈希函数时,开发者往往满足于调用现成的库函数,却错过了理解算…...

别再只输密码了!手把手带你用Wireshark抓包,亲手‘看见’WPA2的四次握手过程(含过滤技巧)

无线网络安全实战:用Wireshark解析WPA2四次握手全流程 家里WiFi突然连不上?明明密码正确却反复提示认证失败?作为网络工程师,我遇到过太多次同事抱怨"路由器抽风",而真相往往藏在那些看不见的数据帧里。今天…...

【奇点2026白皮书核心章节】:为什么83.6%的AI合并失败源于AST抽象层级错配?附可审计的合并决策树V2.1模板

第一章:AST抽象层级错配:AI代码合并失败的根因解构 2026奇点智能技术大会(https://ml-summit.org) AST(Abstract Syntax Tree)是现代AI代码工具理解、生成与重构程序的核心中间表示。然而,当多个AI代理协同执行代码合…...

21天回款950万!拆解素人铺量狂赚的底层逻辑

各位老板、运营操盘手,是不是觉得现在的电商营销越来越像一场豪赌?要么砸重金赌大主播的瞬间爆发,要么烧钱赌信息流的精准推送。结果往往是:流水看起来很猛,利润一算就头疼。今天不聊虚的,我们从市场化运营…...

信息学奥赛一本通 1248:Dungeon Master | 三维迷宫搜索算法精讲

1. 三维迷宫搜索算法入门:从Dungeon Master开始 第一次看到《Dungeon Master》这个题目时,我完全被三维迷宫的概念震撼到了。相比常见的二维迷宫,这个题目要求我们在一个立体的空间中寻找最短路径,就像被困在一个真实的地牢里一样…...

MATLAB条形图进阶:从基础bar函数到数据可视化实战

1. MATLAB条形图基础:从bar函数入门 第一次接触MATLAB的条形图功能时,我也曾被各种参数搞得晕头转向。但后来发现,只要掌握几个核心概念,就能快速上手这个强大的数据可视化工具。bar函数就像是一个万能画笔,能帮我们把…...

Zotero-OCR插件:3步实现PDF文献智能识别与可搜索文本层添加

Zotero-OCR插件:3步实现PDF文献智能识别与可搜索文本层添加 【免费下载链接】zotero-ocr Zotero Plugin for OCR 项目地址: https://gitcode.com/gh_mirrors/zo/zotero-ocr Zotero-OCR是Zotero文献管理软件的功能扩展插件,专门为PDF文档添加可搜索…...

从滤波到优化:手把手拆解VIO算法核心,看懂OpenVINS的MSCKF和ORB-SLAM3的BA到底差在哪

从滤波到优化:手把手拆解VIO算法核心,看懂OpenVINS的MSCKF和ORB-SLAM3的BA到底差在哪 视觉惯性里程计(VIO)技术近年来在机器人导航、增强现实等领域展现出强大的生命力。对于开发者而言,理解不同技术流派的核心差异&am…...

从CH344Q出发:打造高性能USB转4路TTL串口模块的设计实践

1. CH344Q芯片选型与核心优势 第一次接触CH344Q这颗芯片是在去年做一个工业数据采集项目的时候。当时需要同时连接4个不同波特率的传感器设备,市面上常见的USB转串口模块要么速度跟不上,要么稳定性堪忧。折腾了好几款方案后,同事推荐了沁恒的…...

CSS如何实现Less颜色函数自动计算渐变_使用lighten与darken实现视觉反馈

lighten() 和 darken() 按 HSL 的 L 分量线性调整亮度,非像素级明暗处理;需确保输入为 color 类型、慎用于高饱和色、避免链式调用,并配合 saturate 等增强视觉反馈。lighten() 和 darken() 在 Less 中怎么写才不翻车Less 的 lighten() 和 da…...

罗茨风机行业专题研究:全国知名风机大品牌终身有保障的

随着我国工业现代化进程加速及环保政策趋严,罗茨风机作为污水处理、电力、化工等领域的关键设备,市场需求持续增长。据中国通用机械工业协会统计,2024年我国罗茨风机市场规模达82.3亿元,年复合增长率7.5%,行业呈现技术…...

Markdown转docx 保留Latex渲染样式

需求分析 Markdown转docx有大量公式。包括行内公式和行间公式Office 自带的数学与 Latex的渲染样式存在差异。本文倾向于使用Latex的渲染样式轻量调整。转换后仅做轻微的调整,不再编辑公式。 选择方案 主要编辑在Markdown,使用Latex渲染样式&#xff…...

LabVIEW虚拟数字示波器:从仿真到实战的信号分析平台

1. LabVIEW虚拟数字示波器:你的信号分析瑞士军刀 第一次接触LabVIEW虚拟数字示波器时,我正被实验室排队等设备的同学们逼得焦头烂额。直到发现用LabVIEW自己搭建的数字示波器,不仅能随时调用,还能自定义各种高级功能,简…...

最后90天窗口期:2026奇点大会确认的AGI算力-数据-对齐三角瓶颈即将被打破,你的团队准备好了吗?

第一章:2026奇点智能技术大会:通用人工智能最新进展 2026奇点智能技术大会(https://ml-summit.org) 本届大会首次披露多项突破性成果,标志着通用人工智能(AGI)正从理论验证迈入系统化工程实践阶段。来自DeepMind、Op…...

工创赛智能物流搬运小车——V831视觉循迹与STM32色环校准完整方案

工创赛智能物流搬运小车——V831视觉循迹与STM32色环校准完整方案 一、系统概述 1.1 项目背景 本方案针对全国大学生工程实践与创新能力大赛(简称“工创赛”)智能物流搬运赛项,设计并实现基于V831视觉模块的赛道循迹系统,以及基于STM32F4ZGT6的色环校准与运动控制系统。…...

LSTM计算

...

Rescuezilla:系统恢复的瑞士军刀 - 终极免费磁盘克隆与备份解决方案

Rescuezilla:系统恢复的瑞士军刀 - 终极免费磁盘克隆与备份解决方案 【免费下载链接】rescuezilla The Swiss Army Knife of System Recovery 项目地址: https://gitcode.com/gh_mirrors/re/rescuezilla 你是否曾经面对系统崩溃却束手无策?是否需…...

10年老兵带你学Java(第0课):学前必知

Java是什么? Java是一门编程语言,1995年诞生,到现在30年了,依然是全球最流行的语言之一。 打个比方: C语言像手动挡汽车,性能强但难开Python像自动挡汽车,好开但不够快Java像卡车,能…...

FileLocator Pro进阶搜索:从布尔逻辑到正则表达式的实战指南

1. 为什么需要掌握FileLocator Pro的进阶搜索技巧 作为一个常年和各类文档、日志打交道的IT从业者,我深知在海量文件中寻找特定信息有多痛苦。记得有一次,客户服务器突然报错,我需要从几十GB的日志文件中找出特定错误码和关联的请求ID。当时…...

别再死记硬背VGG结构了!手把手教你用PyTorch复现VGG16/19(附代码与权重加载)

从零构建VGG16:PyTorch实战指南与设计哲学解析 在计算机视觉领域,VGG网络以其优雅的对称结构和卓越的性能表现,成为深度学习发展史上的里程碑。不同于简单记忆网络层数,真正理解VGG的精髓在于亲手实现其架构设计。本文将带您用PyT…...

React 大师级思考:如何在不断演变的 Web 标准中保持 React 项目的长期可维护性与扩展性

React 大师级思考:如何在不断演变的 Web 标准中保持 React 项目的长期可维护性与扩展性各位代码的朝圣者,各位在组件海洋中溺水又被救起的勇士们,欢迎来到今天这场关于“如何让 React 项目活过 10 年”的讲座。我知道你们在想什么。你们在想&…...

别再手动算面积和距离了!用Shapely处理GeoJSON数据,效率提升10倍

地理空间数据分析实战:用Shapely解锁GeoJSON处理新姿势 还在用传统方法逐行解析GeoJSON数据?当面对城市地块分析、物流路径优化或区域规划时,手动计算几何属性不仅耗时费力,还容易引入人为误差。这里有一份来自某城市规划局的真实…...

X.509数字证书实战解析:从结构到应用

1. X.509数字证书的前世今生 第一次听说X.509证书时,我正盯着浏览器地址栏那个小锁图标发呆。这个看似简单的技术,实际上支撑着整个互联网的安全通信。X.509就像数字世界的身份证,它用密码学的方式证明了"你是你"。想象一下&#x…...

Qt 6.2 静态编译实战:从环境配置到IDE集成的完整指南

1. 环境准备:搭建静态编译的基础舞台 第一次尝试Qt静态编译时,我盯着满屏的英文文档和报错信息整整发呆了半小时。作为过来人,我理解那种面对复杂工具链的无力感。别担心,跟着我的步骤走,咱们用最稳妥的方式把地基打牢…...

d2s-editor:暗黑破坏神2存档编辑实战指南与深度解析

d2s-editor:暗黑破坏神2存档编辑实战指南与深度解析 【免费下载链接】d2s-editor 项目地址: https://gitcode.com/gh_mirrors/d2/d2s-editor 暗黑破坏神2玩家常面临角色属性不足、装备搭配受限、存档损坏等问题。d2s-editor作为一款基于Web的存档编辑工具&a…...

GHelper轻量级控制工具:三步解决华硕笔记本性能管理难题

GHelper轻量级控制工具:三步解决华硕笔记本性能管理难题 【免费下载链接】g-helper Lightweight, open-source control tool for ASUS laptops and ROG Ally. Manage performance modes, fans, GPU, battery, and RGB lighting across Zephyrus, Flow, TUF, Strix, …...

如何快速掌握LRC歌词制作工具:新手也能上手的完整教程

如何快速掌握LRC歌词制作工具:新手也能上手的完整教程 【免费下载链接】lrc-maker 歌词滚动姬|可能是你所能见到的最好用的歌词制作工具 项目地址: https://gitcode.com/gh_mirrors/lr/lrc-maker 你是否曾经为制作精准的LRC歌词而烦恼&#xff1f…...

如何决定是否需要创建索引_数据区分度与基数Cardinality计算

索引是否有效取决于Cardinality值高低:接近总行数(≥95%)说明区分度高,适合建索引;<10%则单列索引意义不大;低区分度字段应置于联合索引后缀,如(created_at, status),并用…...

Windows右键菜单的“数字园艺师“:ContextMenuManager深度解析与实战手册

Windows右键菜单的"数字园艺师":ContextMenuManager深度解析与实战手册 【免费下载链接】ContextMenuManager 🖱️ 纯粹的Windows右键菜单管理程序 项目地址: https://gitcode.com/gh_mirrors/co/ContextMenuManager 你是否曾想过&…...

在线考试|基于springboot + vue在线考试管理系统(源码+数据库+文档)

在线考试管理系统 目录 基于springboot vue在线考试管理系统 一、前言 二、系统功能演示 详细视频演示 三、技术选型 四、其他项目参考 五、代码参考 六、测试参考 七、最新计算机毕设选题推荐 八、源码获取: 基于springboot vue在线考试管理系统 一、…...