openssl使用哈希算法生成随机密钥
文章目录
- 一、openssl中随机数函数
- **OpenSSL 随机数函数概览**
- 1. **核心随机数函数**
- **常用函数详解**
- 1. `RAND_bytes`
- 2. `RAND_priv_bytes`
- 3. `RAND_seed` 和 `RAND_add`
- 4. `RAND_status`
- **随机数生成器的熵池**
- **常见用例**
- **注意事项**
- 二、使用哈希算法生成随机的密钥
一、openssl中随机数函数
OpenSSL 提供了一套功能强大的随机数生成函数,用于生成高质量的伪随机数。随机数在密码学中至关重要,广泛用于密钥生成、盐值生成、初始化向量 (IV) 等安全操作。
OpenSSL 随机数函数概览
1. 核心随机数函数
| 函数名称 | 描述 |
|---|---|
RAND_bytes | 生成高质量的伪随机字节。 |
RAND_priv_bytes | 生成高质量伪随机字节,推荐用于对安全性要求更高的密钥生成。 |
RAND_pseudo_bytes | 生成伪随机字节(不保证是加密级别安全,已被弃用)。 |
RAND_seed | 手动为伪随机数生成器添加种子值。 |
RAND_add | 添加额外的熵(种子数据)到伪随机数生成器的池中。 |
RAND_status | 检查随机数生成器是否被正确初始化。 |
RAND_poll | 自动收集熵以初始化随机数生成器(由内部调用)。 |
常用函数详解
1. RAND_bytes
-
作用:生成高质量的加密级别伪随机字节。
-
函数原型:
int RAND_bytes(unsigned char *buf, int num);buf:指向用于存储随机字节的缓冲区。num:要生成的随机字节数。- 返回值:
- 成功返回
1。 - 如果随机数生成器未正确初始化,返回
0。
- 成功返回
-
示例:
unsigned char random_bytes[16]; if (RAND_bytes(random_bytes, sizeof(random_bytes)) == 1) {printf("生成的随机数:");for (int i = 0; i < sizeof(random_bytes); i++) {printf("%02x", random_bytes[i]);}printf("\n"); } else {fprintf(stderr, "随机数生成失败!\n"); }
2. RAND_priv_bytes
- 作用:与
RAND_bytes类似,但专为高安全性应用设计(如密钥生成)。可能在实现中增加额外的保护。 - 函数原型:
int RAND_priv_bytes(unsigned char *buf, int num); - 使用方法与
RAND_bytes一致。
3. RAND_seed 和 RAND_add
-
作用:为随机数生成器添加种子值或熵,增强其随机性。
OpenSSL 的随机数生成器依赖熵池,通常会自动初始化,但可以通过这些函数手动添加种子数据。 -
函数原型:
void RAND_seed(const void *buf, int num); void RAND_add(const void *buf, int num, double entropy);buf:种子数据。num:种子数据的字节数。entropy:种子中估计的熵(单位是比特,范围 0 到 8*num)。
-
示例:
unsigned char seed_data[] = {0x12, 0x34, 0x56, 0x78}; RAND_seed(seed_data, sizeof(seed_data));
4. RAND_status
-
作用:检查随机数生成器是否被正确初始化。
-
函数原型:
int RAND_status(void);- 返回值:
- 如果熵池已初始化且可用随机性足够,返回
1。 - 否则返回
0。
- 如果熵池已初始化且可用随机性足够,返回
- 返回值:
-
示例:
if (RAND_status() == 1) {printf("随机数生成器已初始化!\n"); } else {printf("随机数生成器未初始化!\n"); }
随机数生成器的熵池
OpenSSL 的随机数生成器使用熵池作为随机性的来源,依赖于系统提供的随机性(如 /dev/random 或 /dev/urandom)。在大多数情况下,OpenSSL 会自动处理熵池的初始化,但开发者可以通过 RAND_seed 或 RAND_add 提供额外的种子数据。
常见用例
-
生成随机密钥:
unsigned char key[32]; // 256 位密钥 if (RAND_bytes(key, sizeof(key)) == 1) {printf("密钥生成成功!\n"); } -
生成随机初始化向量 (IV):
unsigned char iv[16]; // 128 位 IV if (RAND_bytes(iv, sizeof(iv)) == 1) {printf("IV 生成成功!\n"); } -
增强随机数生成器熵:
unsigned char extra_entropy[] = {0xde, 0xad, 0xbe, 0xef}; RAND_add(extra_entropy, sizeof(extra_entropy), 4.0); // 添加 4 比特熵
注意事项
-
使用
RAND_bytes或RAND_priv_bytes- 推荐使用这两个函数生成随机数,因为它们提供加密级别的安全性。
- 不建议使用
RAND_pseudo_bytes,因为它已被弃用。
-
熵的重要性
- 高质量的熵是随机数生成的核心。如果熵不足,生成的随机数可能会被预测,降低安全性。
-
平台依赖
- OpenSSL 的随机数生成器在不同平台上依赖系统的熵源,例如
/dev/urandom或 Windows 的 CryptGenRandom。
- OpenSSL 的随机数生成器在不同平台上依赖系统的熵源,例如
OpenSSL 的随机数生成函数设计灵活,满足了从简单随机需求到高安全性应用的各种场景。如果需要更高安全性的随机数生成,推荐使用 RAND_priv_bytes。
二、使用哈希算法生成随机的密钥
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <openssl/rand.h>
#include <openssl/evp.h>#define SEED_LENGTH 32 // 随机种子长度(32 字节 = 256 位)
#define KEY_LENGTH 32 // 密钥长度(32 字节 = 256 位)// 使用随机种子和哈希算法生成密钥
void generate_hashed_key(unsigned char *key, size_t key_length) {unsigned char seed[SEED_LENGTH]; // 随机种子// 生成随机种子if (!RAND_bytes(seed, sizeof(seed))) {fprintf(stderr, "随机种子生成失败!\n");exit(EXIT_FAILURE);}// 打印随机种子(调试用)printf("随机种子: ");for (size_t i = 0; i < sizeof(seed); i++) {printf("%02x", seed[i]);}printf("\n");// 使用 SHA-256 哈希函数对种子进行散列EVP_MD_CTX *mdctx = EVP_MD_CTX_new(); // 创建哈希上下文if (!mdctx) {fprintf(stderr, "创建哈希上下文失败!\n");exit(EXIT_FAILURE);}// 初始化哈希计算(使用 SHA-256)if (EVP_DigestInit_ex(mdctx, EVP_sha256(), NULL) != 1) {fprintf(stderr, "哈希初始化失败!\n");EVP_MD_CTX_free(mdctx);exit(EXIT_FAILURE);}// 提供数据进行哈希计算if (EVP_DigestUpdate(mdctx, seed, sizeof(seed)) != 1) {fprintf(stderr, "哈希更新失败!\n");EVP_MD_CTX_free(mdctx);exit(EXIT_FAILURE);}// 获取哈希结果unsigned char hash[EVP_MAX_MD_SIZE];unsigned int hash_len;if (EVP_DigestFinal_ex(mdctx, hash, &hash_len) != 1) {fprintf(stderr, "哈希计算失败!\n");EVP_MD_CTX_free(mdctx);exit(EXIT_FAILURE);}// 释放哈希上下文EVP_MD_CTX_free(mdctx);// 取哈希结果的前 key_length 字节作为密钥if (key_length > hash_len) {fprintf(stderr, "密钥长度超出哈希值长度!\n");exit(EXIT_FAILURE);}memcpy(key, hash, key_length);// 打印生成的密钥(调试用)printf("生成的密钥: ");for (size_t i = 0; i < key_length; i++) {printf("%02x", key[i]);}printf("\n");
}int main() {unsigned char key[KEY_LENGTH]; // 存储生成的密钥// 调用密钥生成函数generate_hashed_key(key, KEY_LENGTH);return 0;
}
运行两次的结果:
这里可以看到每次运行后生成的密钥都是不一样的,这样就保证了密钥的不一致性。

相关文章:
openssl使用哈希算法生成随机密钥
文章目录 一、openssl中随机数函数**OpenSSL 随机数函数概览**1. **核心随机数函数** **常用函数详解**1. RAND_bytes2. RAND_priv_bytes3. RAND_seed 和 RAND_add4. RAND_status **随机数生成器的熵池****常见用例****注意事项** 二、使用哈希算法生成随机的密钥 一、openssl中…...
将word里自带公式编辑器编辑的公式转换成用mathtype编辑的格式
文章目录 将word里自带公式编辑器编辑的公式转换成用mathtype编辑的格式MathType安装问题MathType30天试用延期MathPage.wll文件找不到问题 将word里自带公式编辑器编辑的公式转换成用mathtype编辑的格式 word自带公式编辑器编辑的公式格式: MathType编辑的格式&a…...
校园失物招领系统基于 SpringBoot:点亮校园归还遗失物之光
2系统开发环境 2.1vue技术 Vue (读音 /vjuː/,类似于 view) 是一套用于构建用户界面的渐进式JavaScript框架。 [5] 与其它大型框架不同的是,Vue 被设计为可以自底向上逐层应用。Vue 的核心库只关注视图层,不仅易于上手,还便于与第…...
dhcpd服务器的配置与管理(超详细!!!)
前提条件: (1)虚拟机能够联网(如果nat模式不能联网的看另一期) CentOS7 NAT模式不能联网-CSDN博客 (2)系统是Centos8,因为下载的dhcp-server软件包版本和Centos7不匹配,如果你能成…...
Qml之基本控件
一.Qml常用控件 1.Text(显示普通文本和富文本) 1.1显示普通文本: Window { visible: true width: 320 height: 240 title: qsTr("Hello World") Text { text: "Hello World!" font.family: "Helvetica" font.pointSize: 24 color:…...
【Java从入门到放弃 之 Stream API】
Java Stream API Stream API行为参数化传递代码Lambda表达式Lambda 表达式的语法方法引用 Lambda 表达式的实际应用集合操作并发编程 Lambda 表达式的注意事项总结 Stream API Java8提供了一个全新的API - Stream。引入这个Stream的主要目的,一个是可以支持更好的并…...
Ruby On Rails 笔记1——Rails 入门
突然想跟着官方文档把Ruby On Rails过一遍,把一些有用的记下来就可以一直看了,do它! https://guides.rubyonrails.org/v7.2/ 注:官网是英文文档,我自己翻译了一下,不确保完全准确,只供自己学习开发使用。 …...
高效开发 Python Web 应用:FastAPI 数据验证与响应体设计
高效开发 Python Web 应用:FastAPI 数据验证与响应体设计 目录 🧑💻 FastAPI 的数据验证系统与 Pydantic 模型📦 响应体与模型:定义响应数据的最佳实践🔄 响应模型与查询参数的结合:增强灵活…...
基于“开源 2+1 链动 O2O 商城小程序”的门店拉新策略与流程设计
摘要:在数字化商业浪潮席卷之下,实体门店面临着激烈的市场竞争,如何高效拉新成为关乎门店生存与发展的关键问题。本文聚焦于“开源 21 链动 O2O 商城小程序”,深入探讨结合多种手段的门店拉新策略及详细流程设计。通过剖析到店扫码…...
33.5 remote实战项目之设计prometheus数据源的结构
本节重点介绍 : 项目要求 通过remote read读取prometheus中的数据通过remote write向prometheus中写入数据 准备工作 新建项目 prome_remote_read_write设计prometheus 数据源的结构初始化 项目要求 通过remote read读取prometheus中的数据通过remote write向prometheus中写…...
微服务springboot详细解析(一)
目录 1.Spring概述 2.什么是SpringBoot? 3.第一个SpringBoot程序 4.配置参数优先级 5.springboot自动装配原理 6.SpringBootApplication&SpringApplication.run 7.ConfigurationProperties(prefix "") 8.Validated数据校验 29、聊聊该如何写一…...
深入探讨Go语言中的双向链表
简介 双向链表是链表家族中的一种高级结构,每个节点不仅指向下一个节点,还指向上一个节点。今天,我们将学习如何在Go语言中实现和操作这种灵活的数据结构。 双向链表的优缺点 优点: 可以从任一方向遍历链表,灵活性高…...
Fastapi + vue3 自动化测试平台---移动端App自动化篇
概述 好久写文章了,专注于新框架,新UI界面的实践,废话不多说,开搞 技术架构 后端: Fastapi Airtest multiprocessing 前端: 基于 Vue3、Vite、TypeScript、Pinia、Pinia持久化插件、Unocss 和 Elemen…...
ElasticSearch easy-es 聚合函数 group by 混合写法求Top N 词云 分词
1.将用户访问记录表数据同步到ES,并且分词,获取用户访问最多前十条词语。 Elasticsearch、Easy-es 快速入门 SearchAfterPage分页 若依前后端分离 Ruoyi-Vue SpringBoot 使用结巴分词器 <!-- 分词器--><dependency><groupId>com.hua…...
在 ASP.NET C# Web API 中实现 Serilog 以增强请求和响应的日志记录
介绍 日志记录是任何 Web 应用程序的关键方面。它有助于调试、性能监控和了解用户交互。在 ASP.NET C# 中,集成 Serilog 作为记录请求和响应(包括传入和传出的数据)的中间件可以显著提高 Web API 的可观察性和故障排除能力。 在过去的几周里&…...
2024年顶级小型语言模型前15名
本文,我们将深入了解2024年备受瞩目的十五款小型语言模型(SLMs),它们分别是Llama 3.1 8B、Gemma2、Qwen 2、Mistral Nemo、Phi-3.5等。这些SLMs以其精巧的体积和高效率著称,它们不需要依赖庞大的服务器资源,…...
精通 Python 网络安全(一)
前言 最近,Python 开始受到越来越多的关注,最新的 Python 更新添加了许多可用于执行关键任务的包。我们的主要目标是帮助您利用 Python 包来检测和利用漏洞,并解决网络挑战。 本书将首先带您了解与网络和安全相关的 Python 脚本和库。然后&…...
【python自动化二】pytest集成allure生成测试报告
pytest本身不会直接生成测试报告,而allure是一种生成测试报告的公共插件,可与多种测试框架配合生成测试报告,本文介绍下如何集成allure生成测试报告。 1.allure安装 1.安装allure-pytest 先安装allure的pytest插件,用于在pytes…...
网络版本的通讯录青春版(protobuf)
环境搭建 Protobuf 还常⽤于通讯协议、服务端数据交换场景。 因为我们主要目的只是为了学习protobuf,因此对于客户端,原本应该具备: 新增⼀个联系⼈ ◦ 删除⼀个联系⼈ ◦ 查询通讯录列表 ◦ 查询⼀个联系⼈的详细信息 这样四个功能。 …...
开源模型应用落地-安全合规篇-用户输入价值观判断(三)
一、前言 在深度合规功能中,对用户输入内容的价值观判断具有重要意义。这一功能不仅仅是对信息合法性和合规性的简单审核,更是对信息背后隐含的伦理道德和社会责任的深刻洞察。通过对价值观的判断,系统能够识别可能引发不当影响或冲突的内容,从而为用户提供更安全、更和谐的…...
Vue记事本应用实现教程
文章目录 1. 项目介绍2. 开发环境准备3. 设计应用界面4. 创建Vue实例和数据模型5. 实现记事本功能5.1 添加新记事项5.2 删除记事项5.3 清空所有记事 6. 添加样式7. 功能扩展:显示创建时间8. 功能扩展:记事项搜索9. 完整代码10. Vue知识点解析10.1 数据绑…...
椭圆曲线密码学(ECC)
一、ECC算法概述 椭圆曲线密码学(Elliptic Curve Cryptography)是基于椭圆曲线数学理论的公钥密码系统,由Neal Koblitz和Victor Miller在1985年独立提出。相比RSA,ECC在相同安全强度下密钥更短(256位ECC ≈ 3072位RSA…...
【机器视觉】单目测距——运动结构恢复
ps:图是随便找的,为了凑个封面 前言 在前面对光流法进行进一步改进,希望将2D光流推广至3D场景流时,发现2D转3D过程中存在尺度歧义问题,需要补全摄像头拍摄图像中缺失的深度信息,否则解空间不收敛…...
【ROS】Nav2源码之nav2_behavior_tree-行为树节点列表
1、行为树节点分类 在 Nav2(Navigation2)的行为树框架中,行为树节点插件按照功能分为 Action(动作节点)、Condition(条件节点)、Control(控制节点) 和 Decorator(装饰节点) 四类。 1.1 动作节点 Action 执行具体的机器人操作或任务,直接与硬件、传感器或外部系统…...
React19源码系列之 事件插件系统
事件类别 事件类型 定义 文档 Event Event 接口表示在 EventTarget 上出现的事件。 Event - Web API | MDN UIEvent UIEvent 接口表示简单的用户界面事件。 UIEvent - Web API | MDN KeyboardEvent KeyboardEvent 对象描述了用户与键盘的交互。 KeyboardEvent - Web…...
华为OD机试-食堂供餐-二分法
import java.util.Arrays; import java.util.Scanner;public class DemoTest3 {public static void main(String[] args) {Scanner in new Scanner(System.in);// 注意 hasNext 和 hasNextLine 的区别while (in.hasNextLine()) { // 注意 while 处理多个 caseint a in.nextIn…...
Java面试专项一-准备篇
一、企业简历筛选规则 一般企业的简历筛选流程:首先由HR先筛选一部分简历后,在将简历给到对应的项目负责人后再进行下一步的操作。 HR如何筛选简历 例如:Boss直聘(招聘方平台) 直接按照条件进行筛选 例如:…...
docker 部署发现spring.profiles.active 问题
报错: org.springframework.boot.context.config.InvalidConfigDataPropertyException: Property spring.profiles.active imported from location class path resource [application-test.yml] is invalid in a profile specific resource [origin: class path re…...
python报错No module named ‘tensorflow.keras‘
是由于不同版本的tensorflow下的keras所在的路径不同,结合所安装的tensorflow的目录结构修改from语句即可。 原语句: from tensorflow.keras.layers import Conv1D, MaxPooling1D, LSTM, Dense 修改后: from tensorflow.python.keras.lay…...
STM32HAL库USART源代码解析及应用
STM32HAL库USART源代码解析 前言STM32CubeIDE配置串口USART和UART的选择使用模式参数设置GPIO配置DMA配置中断配置硬件流控制使能生成代码解析和使用方法串口初始化__UART_HandleTypeDef结构体浅析HAL库代码实际使用方法使用轮询方式发送使用轮询方式接收使用中断方式发送使用中…...
