深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法
深入浅出:JavaScript 中的 window.crypto.getRandomValues()
方法
在现代 Web 开发中,随机数的生成看似简单,却隐藏着许多玄机。无论是生成密码、加密密钥,还是创建安全令牌,随机数的质量直接关系到系统的安全性。JavaScript 提供了多种生成随机数的方法,而其中 window.crypto.getRandomValues()
是最值得开发者关注的高安全方案。本文将深入解析这一方法的原理、用法和适用场景,帮助你更好地理解其背后的技术逻辑。
一、为什么需要加密级的随机数?
随机数在计算机领域无处不在,但并非所有随机数都“生而平等”。常见的 Math.random()
方法虽然能生成随机数,但它基于伪随机数生成算法(PRNG),其输出结果在理论上是可预测的。例如,如果你知道算法的种子(seed)和内部状态,就能推算出后续的随机数。
而在密码学、安全认证等场景中,随机数的不可预测性至关重要。如果随机数被攻击者猜到,可能会导致密钥泄露、会话令牌被破解等严重后果。因此,JavaScript 引入了 Web Crypto API,其中的 window.crypto.getRandomValues()
方法通过调用操作系统的加密安全随机数生成器(CSPRNG),确保生成的随机数具备更高的安全性和不可预测性。
二、window.crypto.getRandomValues()
的原理
1. 什么是 CSPRNG?
CSPRNG(Cryptographically Secure Pseudorandom Number Generator)是一种专门设计用于密码学的伪随机数生成器。它的核心特点包括:
- 高熵源:CSPRNG 依赖于操作系统的熵池(entropy pool),这些熵通常来自硬件层面的随机事件(如键盘敲击时间、鼠标移动轨迹、磁盘 I/O 延迟等)。
- 不可预测性:即使知道之前的输出,也无法推算出后续的随机数。
- 抗攻击性:经过严格的密码学验证,确保输出结果不会被外部干扰。
在浏览器中,window.crypto.getRandomValues()
本质上是对系统级 CSPRNG 的封装。例如,在 Linux 系统中,它可能调用 /dev/urandom
;在 Windows 中,它可能使用 CryptGenRandom
或 BCryptGenRandom
。
2. TypedArray 的作用
window.crypto.getRandomValues()
需要传入一个 TypedArray 类型的参数(如 Uint8Array
、Uint16Array
、Uint32Array
等)。这是因为加密级随机数通常以字节(byte)为单位生成,而 TypedArray 可以高效地处理二进制数据。
例如,生成一个包含 10 个 8 位无符号整数的数组:
const array = new Uint8Array(10);
window.crypto.getRandomValues(array);
console.log(array); // 输出类似 [234, 56, 12, 78, ...]
三、如何正确使用 window.crypto.getRandomValues()
?
1. 生成随机字节数组
这是最常见的用法,适用于需要原始随机字节的场景:
const randomBytes = new Uint8Array(16); // 生成 16 字节的随机数
window.crypto.getRandomValues(randomBytes);
console.log(randomBytes); // 输出类似 [123, 45, 67, 89, ...]
2. 生成指定范围的随机整数
如果需要生成某个范围内的随机整数(如 0-100),可以通过对随机数进行取模运算:
function getRandomInt(min, max) {const range = max - min;const array = new Uint32Array(1);window.crypto.getRandomValues(array);return Math.floor((array[0] / 4294967296) * range) + min;
}console.log(getRandomInt(0, 100)); // 输出 0-100 之间的随机整数
3. 生成十六进制字符串
在生成唯一标识符(如 UUID)或哈希值时,十六进制字符串非常有用:
function randomHex(length) {const array = new Uint8Array(length / 2); // 每个字节对应 2 位十六进制字符window.crypto.getRandomValues(array);return Array.from(array, byte => ('0' + byte.toString(16)).slice(-2)).join('');
}console.log(randomHex(16)); // 输出类似 "a1b2c3d4e5f67890"
4. 生成随机密码
结合字符集和随机数,可以生成强随机密码:
function generatePassword(length) {const charset = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!@#$%^&*';const values = new Uint8Array(length);window.crypto.getRandomValues(values);let password = '';for (let i = 0; i < length; i++) {password += charset[values[i] % charset.length];}return password;
}console.log(generatePassword(12)); // 输出类似 "A3$fG7!kL2@q"
四、适用场景
1. 密码学应用
- 生成加密密钥(如 AES、RSA)。
- 创建初始化向量(IV)或盐值(Salt)。
- 加密通信协议中的随机数生成(如 TLS 握手)。
2. 安全敏感场景
- 生成一次性令牌(One-Time Token)。
- 验证用户身份的随机挑战(Challenge)。
- 安全审计日志中的唯一标识符。
3. 游戏与抽奖系统
- 在需要公平性的场景中生成不可预测的结果(如彩票开奖、游戏道具掉落率)。
五、注意事项
1. 浏览器兼容性
window.crypto.getRandomValues()
在现代浏览器(Chrome、Firefox、Edge、Safari)中均得到良好支持。但在旧版浏览器(如 IE)中可能不可用。建议在使用前进行兼容性检查:
if (window.crypto && window.crypto.getRandomValues) {// 支持加密级随机数生成
} else {// 回退到 Math.random() 或提示用户升级浏览器
}
2. 性能与成本
CSPRNG 的生成速度通常低于普通 PRNG(如 Math.random()
)。在需要大量随机数的非安全场景(如动画效果、游戏关卡生成)中,优先使用 Math.random()
。
3. 避免误用
- 不要用于生成低熵值:例如,如果只需要一个布尔值,直接使用
Math.random() > 0.5
更高效。 - 不要混淆加密与非加密需求:在安全场景中必须使用
window.crypto.getRandomValues()
,而在普通场景中使用Math.random()
即可。
六、总结
window.crypto.getRandomValues()
是 JavaScript 中生成加密安全随机数的核心方法。它通过调用系统级 CSPRNG,确保随机数的不可预测性和高安全性。无论是密码学、安全认证,还是需要公平性的应用场景,开发者都应优先考虑使用这一方法。
然而,技术的选择始终需要权衡。在非安全场景中,Math.random()
依然是更高效的方案。理解两者的差异,并根据需求合理选择,才是开发者真正的“安全之道”。
附录:常见问题解答
Q1: window.crypto.getRandomValues()
和 Math.random()
有什么区别?
- 安全性:
window.crypto.getRandomValues()
使用 CSPRNG,而Math.random()
使用 PRNG。 - 性能:
Math.random()
生成速度更快,但不适合安全场景。
Q2: 为什么不能直接用 window.crypto.getRandomValues()
生成 0-1 之间的浮点数?
因为 CSPRNG 生成的是整数,需要通过除法运算将其转换为浮点数(如 value / 2^32
)。
Q3: 如果在 Node.js 中使用 window.crypto.getRandomValues()
会怎样?
window.crypto
是浏览器端的 API,在 Node.js 中不可用。Node.js 提供了 crypto
模块(如 crypto.randomBytes()
)来实现类似功能。
希望这篇文章能帮助你更好地理解和应用 window.crypto.getRandomValues()
。在开发过程中,始终记住:安全无小事,随机数的生成方式可能决定系统的生死存亡。
相关文章:
深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法
深入浅出:JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中,随机数的生成看似简单,却隐藏着许多玄机。无论是生成密码、加密密钥,还是创建安全令牌,随机数的质量直接关系到系统的安全性。Jav…...
【SpringBoot】100、SpringBoot中使用自定义注解+AOP实现参数自动解密
在实际项目中,用户注册、登录、修改密码等操作,都涉及到参数传输安全问题。所以我们需要在前端对账户、密码等敏感信息加密传输,在后端接收到数据后能自动解密。 1、引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId...
【位运算】消失的两个数字(hard)
消失的两个数字(hard) 题⽬描述:解法(位运算):Java 算法代码:更简便代码 题⽬链接:⾯试题 17.19. 消失的两个数字 题⽬描述: 给定⼀个数组,包含从 1 到 N 所有…...

无法与IP建立连接,未能下载VSCode服务器
如题,在远程连接服务器的时候突然遇到了这个提示。 查阅了一圈,发现是VSCode版本自动更新惹的祸!!! 在VSCode的帮助->关于这里发现前几天VSCode自动更新了,我的版本号变成了1.100.3 才导致了远程连接出…...
STM32+rt-thread判断是否联网
一、根据NETDEV_FLAG_INTERNET_UP位判断 static bool is_conncected(void) {struct netdev *dev RT_NULL;dev netdev_get_first_by_flags(NETDEV_FLAG_INTERNET_UP);if (dev RT_NULL){printf("wait netdev internet up...");return false;}else{printf("loc…...

基于Flask实现的医疗保险欺诈识别监测模型
基于Flask实现的医疗保险欺诈识别监测模型 项目截图 项目简介 社会医疗保险是国家通过立法形式强制实施,由雇主和个人按一定比例缴纳保险费,建立社会医疗保险基金,支付雇员医疗费用的一种医疗保险制度, 它是促进社会文明和进步的…...

SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...
《Playwright:微软的自动化测试工具详解》
Playwright 简介:声明内容来自网络,将内容拼接整理出来的文档 Playwright 是微软开发的自动化测试工具,支持 Chrome、Firefox、Safari 等主流浏览器,提供多语言 API(Python、JavaScript、Java、.NET)。它的特点包括&a…...

vscode(仍待补充)
写于2025 6.9 主包将加入vscode这个更权威的圈子 vscode的基本使用 侧边栏 vscode还能连接ssh? debug时使用的launch文件 1.task.json {"tasks": [{"type": "cppbuild","label": "C/C: gcc.exe 生成活动文件"…...
可靠性+灵活性:电力载波技术在楼宇自控中的核心价值
可靠性灵活性:电力载波技术在楼宇自控中的核心价值 在智能楼宇的自动化控制中,电力载波技术(PLC)凭借其独特的优势,正成为构建高效、稳定、灵活系统的核心解决方案。它利用现有电力线路传输数据,无需额外布…...
Qt Widget类解析与代码注释
#include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this); }Widget::~Widget() {delete ui; }//解释这串代码,写上注释 当然可以!这段代码是 Qt …...

解决Ubuntu22.04 VMware失败的问题 ubuntu入门之二十八
现象1 打开VMware失败 Ubuntu升级之后打开VMware上报需要安装vmmon和vmnet,点击确认后如下提示 最终上报fail 解决方法 内核升级导致,需要在新内核下重新下载编译安装 查看版本 $ vmware -v VMware Workstation 17.5.1 build-23298084$ lsb_release…...
线程与协程
1. 线程与协程 1.1. “函数调用级别”的切换、上下文切换 1. 函数调用级别的切换 “函数调用级别的切换”是指:像函数调用/返回一样轻量地完成任务切换。 举例说明: 当你在程序中写一个函数调用: funcA() 然后 funcA 执行完后返回&…...

(二)TensorRT-LLM | 模型导出(v0.20.0rc3)
0. 概述 上一节 对安装和使用有个基本介绍。根据这个 issue 的描述,后续 TensorRT-LLM 团队可能更专注于更新和维护 pytorch backend。但 tensorrt backend 作为先前一直开发的工作,其中包含了大量可以学习的地方。本文主要看看它导出模型的部分&#x…...

CMake基础:构建流程详解
目录 1.CMake构建过程的基本流程 2.CMake构建的具体步骤 2.1.创建构建目录 2.2.使用 CMake 生成构建文件 2.3.编译和构建 2.4.清理构建文件 2.5.重新配置和构建 3.跨平台构建示例 4.工具链与交叉编译 5.CMake构建后的项目结构解析 5.1.CMake构建后的目录结构 5.2.构…...

visual studio 2022更改主题为深色
visual studio 2022更改主题为深色 点击visual studio 上方的 工具-> 选项 在选项窗口中,选择 环境 -> 常规 ,将其中的颜色主题改成深色 点击确定,更改完成...

关于nvm与node.js
1 安装nvm 安装过程中手动修改 nvm的安装路径, 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解,但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后,通常在该文件中会出现以下配置&…...
FastAPI 教程:从入门到实践
FastAPI 是一个现代、快速(高性能)的 Web 框架,用于构建 API,支持 Python 3.6。它基于标准 Python 类型提示,易于学习且功能强大。以下是一个完整的 FastAPI 入门教程,涵盖从环境搭建到创建并运行一个简单的…...
java调用dll出现unsatisfiedLinkError以及JNA和JNI的区别
UnsatisfiedLinkError 在对接硬件设备中,我们会遇到使用 java 调用 dll文件 的情况,此时大概率出现UnsatisfiedLinkError链接错误,原因可能有如下几种 类名错误包名错误方法名参数错误使用 JNI 协议调用,结果 dll 未实现 JNI 协…...

使用分级同态加密防御梯度泄漏
抽象 联邦学习 (FL) 支持跨分布式客户端进行协作模型训练,而无需共享原始数据,这使其成为在互联和自动驾驶汽车 (CAV) 等领域保护隐私的机器学习的一种很有前途的方法。然而,最近的研究表明&…...
基于服务器使用 apt 安装、配置 Nginx
🧾 一、查看可安装的 Nginx 版本 首先,你可以运行以下命令查看可用版本: apt-cache madison nginx-core输出示例: nginx-core | 1.18.0-6ubuntu14.6 | http://archive.ubuntu.com/ubuntu focal-updates/main amd64 Packages ng…...

理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端
🌟 什么是 MCP? 模型控制协议 (MCP) 是一种创新的协议,旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议,它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...

iPhone密码忘记了办?iPhoneUnlocker,iPhone解锁工具Aiseesoft iPhone Unlocker 高级注册版分享
平时用 iPhone 的时候,难免会碰到解锁的麻烦事。比如密码忘了、人脸识别 / 指纹识别突然不灵,或者买了二手 iPhone 却被原来的 iCloud 账号锁住,这时候就需要靠谱的解锁工具来帮忙了。Aiseesoft iPhone Unlocker 就是专门解决这些问题的软件&…...
AtCoder 第409场初级竞赛 A~E题解
A Conflict 【题目链接】 原题链接:A - Conflict 【考点】 枚举 【题目大意】 找到是否有两人都想要的物品。 【解析】 遍历两端字符串,只有在同时为 o 时输出 Yes 并结束程序,否则输出 No。 【难度】 GESP三级 【代码参考】 #i…...
Python爬虫实战:研究feedparser库相关技术
1. 引言 1.1 研究背景与意义 在当今信息爆炸的时代,互联网上存在着海量的信息资源。RSS(Really Simple Syndication)作为一种标准化的信息聚合技术,被广泛用于网站内容的发布和订阅。通过 RSS,用户可以方便地获取网站更新的内容,而无需频繁访问各个网站。 然而,互联网…...

【网络安全产品大调研系列】2. 体验漏洞扫描
前言 2023 年漏洞扫描服务市场规模预计为 3.06(十亿美元)。漏洞扫描服务市场行业预计将从 2024 年的 3.48(十亿美元)增长到 2032 年的 9.54(十亿美元)。预测期内漏洞扫描服务市场 CAGR(增长率&…...

DAY 47
三、通道注意力 3.1 通道注意力的定义 # 新增:通道注意力模块(SE模块) class ChannelAttention(nn.Module):"""通道注意力模块(Squeeze-and-Excitation)"""def __init__(self, in_channels, reduction_rat…...

Linux相关概念和易错知识点(42)(TCP的连接管理、可靠性、面临复杂网络的处理)
目录 1.TCP的连接管理机制(1)三次握手①握手过程②对握手过程的理解 (2)四次挥手(3)握手和挥手的触发(4)状态切换①挥手过程中状态的切换②握手过程中状态的切换 2.TCP的可靠性&…...

UE5 学习系列(三)创建和移动物体
这篇博客是该系列的第三篇,是在之前两篇博客的基础上展开,主要介绍如何在操作界面中创建和拖动物体,这篇博客跟随的视频链接如下: B 站视频:s03-创建和移动物体 如果你不打算开之前的博客并且对UE5 比较熟的话按照以…...

【JVM】- 内存结构
引言 JVM:Java Virtual Machine 定义:Java虚拟机,Java二进制字节码的运行环境好处: 一次编写,到处运行自动内存管理,垃圾回收的功能数组下标越界检查(会抛异常,不会覆盖到其他代码…...