RSA的原理和简单实践
RSA加密是一种非对称加密,原理是:
- 使⽤算法可以⽣成两把钥匙 A 和 B
- 使⽤ A 加密的信息,使⽤ B 可以解开
- 使⽤ B 加密的信息,使⽤ A 可以解开
⽇常使⽤中,我们把⼀把作为公钥公开发布。⼀把作为私钥,⾃⼰保留。这样,任何⼈都可以使⽤我们的公钥加密信息发给我们,我们则可以使⽤⾃⼰的私钥解开。
只要把私钥保存好,这个通信系统就⾮常安全。
数学基础
1. 欧拉函数
欧拉函数的输入是一个正整数,输出小于这个正整数的、跟它互质的整数数量。
定义是:
\]
\(p_1, p_2, \cdots, p_m\) 表示 \(n\) 的 $ m $ 个质因子,重复的算一个。
比如5,它是个质数,只有它自己一个因子,所以\(\phi(5) = 5 \times (1 - \frac{1}{5}) = 4\)。
当然你也能猜出来,因为跟一个质数互质的数就是从1到它的前驱数的全部自然数,所以对于任意质数 \(p\) 都有
\]
对于合数呢?比如 \(6 = 2 \times 3\),所以 \(\phi(6) = 6 \times \frac{1}{2} \times \frac{2}{3} = 2\) ,也就是6有2个互质数,我们数一下就是1和5。
再看一个比如12,\(12 = 2^2 \times 3\),所以 \(\phi(12) = 12 \times \frac{1}{2} \times \frac{2}{3} = 4\)。我们数一下12的互质数有1、5、7、11四个。
规定\(\phi(1)=1\)
欧拉函数的证明比较简单,可以自行AI。
2. 欧拉定理
当正整数 \(a\) 和 \(n\) 互质时,有
\]
换句话说,\(a^{\phi(n)} - 1\) 可以被 \(n\) 整除。
例如,7和10互质。\(7^{\phi(10)} = 7^4 = 2401\),减去1时10的倍数;
反过来,\(10^{\phi(7)} = 10^6 = 100,0000\),减去1是999999,\(999999 \div 7 = 142857\) (就是1/7 的循环节)。
3. 模反元素
从上面计算的过程可以看出来,如果正整数 \(a\) 和 \(n\) 互质时,一定能找出一个正整数 \(b\),使得
\]
\(b\) 就叫 \(a\) 的模反元素。
模反元素肯定存在。最起码,由于 \(a^{\phi(n)} = a a^{\phi(n) - 1} \equiv 1 (\textbf{mod } n)\) 所以 $ b = a^{\phi(n) - 1}$。
实际上,\(b\) 加减 \(n\) 的倍数都是 \(a\) 的模反元素。
比如上面看到了,10对7的模反元素可以是10万,因为100万减1是7的倍数。那么用10万减去7的14285倍也就是99995得到5,5乘以10得到50,再减1也是7的倍数。
密钥生成
上面就是全部的数学基础。通过这些可以来生成密钥了。
1. 随机选择两个大质数p和q并计算他们的积n
为了演示,这里选择p = 7和q = 11,有 n = 77。
实际应用中,要求n的位数在600位以上才能保证安全。
因为要求n的二进制位大于2048,折成十进制就是617位以上
2. 计算n的欧拉函数 \(\phi(n)\)
虽然n很大,但是由于p和q是质数,所以就简单了
\]
在我们例子里就是6*10 = 60。
后面为了写起来方便,用字母 \(z\) 表示欧拉函数的结果:\(z = \phi(n)\)。
3. 选择一个数e跟 \(\phi(n)\) 互质并计算e的模反元素d
要找一个数跟 \(z\) 互质,e可以比 \(\phi(n)\) 更大。不过 \(\phi(n)\) 已经很大了,所以一般最大也就使用65537。
使用公开的数不会降低系统安全性
这里需要跟60互质,我们选择e = 13。
简单应用上面的方法,可以得到 \(d = e^{\phi({60})-1}\)。
这个60比较小,\(60=2^2 \times 3 \times 5\), 所以\(\phi(60) = 16\), \(d = 13 ^ {15}\) 虽然大但是也能算出来。
但是在实际中,\(z\) 通常大得很,就算能求出它的欧拉函数值,模反元素也算不出来。
扩展欧几里得算法
换一种思路。既然 \(ed \equiv 1 (\textbf{mod } z)\),也就是 \(ed-kz=1\),k是某个整数。
而扩展欧几里得算法不仅可以求出两个整数a和b的最大公约数d,还可以找到整数x和y,使得
\]
算法实现很简单:
fn extended_gcd(a: i64, b: i64) -> (i64, i64, i64) {
if b == 0 {
(a, 1, 0)
} else {
let (gcd, x1, y1) = extended_gcd(b, a % b);
let x = y1;
let y = x1 - (a / b) * y1;
(gcd, x, y)
}
}
代入13和60,得到d=-23。给它加60的倍数使它变成正数,所以d=37。
这样,公钥就是[n, e]=[77,13],私钥就是[n,d]=[77,37]。
私钥的安全性
已知公钥能算出私钥吗?
- 因为 \(ed \equiv 1 (\textbf{mod } z)\),而e已知,所以想算出d需要知道z
- \(z = \phi(n) = (p -1)(q-1)\),需要拿到p和q
- \(n = p \times q\),n已知,分解质因数可得p和q
所以这套逻辑的保证就是第三步很难。而一旦成功分解了,私钥就很容易算了。
加密和解密
加密过程
被加密的消息 m 需要是⼀个⼩于 n 的整数(我们可以将任意字节流直接解读为⼀个⽆符号整数)。如果消息太⼤,解读为整数以后⽐ n 要⼤,那么就分段加密。
实际应用中,我们不会直接⽤ RSA 来加密消息,⽽是⽤ RSA 来加密⼀个对称秘钥,再⽤这个秘钥加密消息。
加密的过程就是计算下面这个c的过程:
\]
假设我们要加密的消息是50,使用上面的计算
\]
可得c=29,这就是加密后的消息。
计算c的算法可以参考
fn main() {
let base = 50; // 原始消息,不能大于77
let exponent = 13;
let modulus = 77;
let result = modular_exponentiation(base, exponent, modulus);
println!("{}", result); // 加密结果
}
fn modular_exponentiation(base: i64, exponent: i64, modulus: i64) -> i64 {
let mut result = 1;
let mut base = base % modulus;
let mut exponent = exponent;
while exponent > 0 {
if exponent % 2 == 1 {
result = (result * base) % modulus;
}
exponent >>= 1;
base = (base * base) % modulus;
}
result
}
解密过程
解密过程是一样的,依据是
\]
let base = 29;
let exponent = 37;
let modulus = 77;
let result = modular_exponentiation(base, exponent, modulus);
println!("{}", result);
输出结果是50,就是我们原来的消息。
解密依据的证明
为什么当 \(m^{e} \equiv c (\textbf{ mod } n)\) 时会有 \(c^{d} \equiv m (\textbf{ mod } n)\) 呢?
\therefore c= m^e-kn
\]
代入目标式:
\]
根据二项式定理,左边展开后除了第一项是$ m^{ed}$ 其余项都含有 \(kn\),必然是n的倍数,所以舍弃这些项。只要证明
\]
即可。
根据定义,
\therefore ed = 1+hz
\]
代入可得
\]
- 当m和n互质时
\therefore m^{\phi(n)} = rn + 1 \\
m^{\phi(n)h} = (rn + 1)^h \overset{\underset{\mathrm{二项式定理}}{}}{=} tn + 1 \\
\therefore m^{\phi(n)h} \equiv 1(\textbf{ mod } n) \\
\therefore m^{\phi(n)h} m \equiv m(\textbf{ mod } n)
\]
得证。
2. 当m和n不互质
不互质时m只能时p或q的倍数。
以 \(m = kp\) 为例,k必然小于q,因为m<n。因为q是质数,所以k与q互质。同时m也跟q互质,否则m就大于n了。
\therefore (kp)^{q-1} \equiv 1(\textbf{ mod } q) \\
\therefore (kp)^{q-1} =tq + 1 \\
\therefore (kp)^{(q-1)h(p-1)} =(tq + 1)^{h(p-1)}
\]
又根据二项式定理
\therefore (kp)^{(q-1)h(p-1)} kp \equiv kp (\textbf{ mod } q) \\
(kp)^{ed} \equiv kp (\textbf{ mod } q) \\
\therefore (kp)^{ed} = kp + tq
\]
要使等式成立,每一项都需要是p的倍数,所以tq是p的倍数。因为q不是p的倍数,所以t是p的倍数 t = vp:
m^{ed} = m + vn \\
\therefore m^{ed} \equiv m (\textbf{ mod } n)
\]
得证。
RSA的原理和简单实践的更多相关文章
- Spring 学习二-----AOP的原理与简单实践
一.Spring AOP的原理 AOP全名Aspect-Oriented Programming,中文直译为面向切面(方面)编程.何为切面,就比如说我们系统中的权限管理,日志,事务等我们都可以将其看 ...
- 带你彻底理解RSA算法原理,很简单的
1. 什么是RSA RSA算法是现今使用最广泛的公钥密码算法,也是号称地球上最安全的加密算法. 在了解RSA算法之前,先熟悉下几个术语 根据密钥的使用方法,可以将密码分为 对称密码 和 公钥密码 对称 ...
- RSA算法原理(简单易懂)
1. 什么是RSA RSA算法是现今使用最广泛的公钥密码算法,也是号称地球上最安全的加密算法.在了解RSA算法之前,先熟悉下几个术语 根据密钥的使用方法,可以将密码分为对称密码和公钥密码 对称密码:加 ...
- kafka原理和实践(二)spring-kafka简单实践
系列目录 kafka原理和实践(一)原理:10分钟入门 kafka原理和实践(二)spring-kafka简单实践 kafka原理和实践(三)spring-kafka生产者源码 kafka原理和实践( ...
- 轻松学习RSA加密算法原理
转自:http://blog.csdn.net/sunmenggmail/article/details/11994013 http://blog.csdn.net/q376420785/articl ...
- 轻松学习RSA加密算法原理 (转)
轻松学习RSA加密算法原理 (转) http://blog.csdn.net/q376420785/article/details/8557266 http://www.ruanyifeng.com/ ...
- Tengine HTTPS原理解析、实践与调试【转】
本文邀请阿里云CDN HTTPS技术专家金九,分享Tengine的一些HTTPS实践经验.内容主要有四个方面:HTTPS趋势.HTTPS基础.HTTPS实践.HTTPS调试. 一.HTTPS趋势 这一 ...
- RSA加密算法原理及RES签名算法简介
第一部分:RSA算法原理与加密解密 一.RSA加密过程简述 A和B进行加密通信时,B首先要生成一对密钥.一个是公钥,给A,B自己持有私钥.A使用B的公钥加密要加密发送的内容,然后B在通过自己的私钥解密 ...
- SSH原理与运用(一)和(二):远程登录 RSA算法原理(一)和(二)
SSH原理与运用(一)和(二):远程登录 RSA算法原理(一)和(二) http://www.ruanyifeng.com/blog/2011/12/ssh_remote_login.html ht ...
- MapReduce 原理与 Python 实践
MapReduce 原理与 Python 实践 1. MapReduce 原理 以下是个人在MongoDB和Redis实际应用中总结的Map-Reduce的理解 Hadoop 的 MapReduce ...
随机推荐
- 4G模组PSM+超低功耗,手把手教你!
合宙4G-Cat.1模组支持三种功耗模式: 常规模式 低功耗模式 PSM+模式 用户可以根据不同的应用场景,按需选择不同的合宙4G-Cat.1模组功耗模式,以及三种功耗模式之间的相互转换. 合宙 ...
- 是时候放弃Scratch了,你有更好的扣叮coding
原创 IT软件部落 IT软件部落 Scratch是由麻省理工学院媒体实验室开发的,拥有一个庞大的全球社区,用户可以在社区中分享和交流自己的作品,适用于全球各个领域,包括学校.社区和个人等. 可是你有没 ...
- MinIO Linux 安装使用 & SpringBoot整合MinIO
目录 MinIO Linux 安装 单节点部署 创建 systemd 系统启动服务文件 创建环境变量文件 启动MinIO服务 连接到MinIO服务 SpringBoot项目整合MinIO 配置项 工具 ...
- linux下文件夹文件名称最大长度
今天突发奇想,如果创建一个文件,不写入内容,就如我们之前说的写入扩展属性能快速查找数据,但是在SSD下只能写4000个左右的字符,那么有没有更快速的方法存储这样的信息呢? 我想到可以同文件名来存储信息 ...
- JConsole 远程监控Tomcat服务
JConsole 远程监控Tomcat服务 1.概述 JConsole是一个基于JMX的GUI工具,用于连接正在运行的JVM,不过此JVM需要使用可管理的模式启动.如果要把一个应用以可管理的形式启动, ...
- 导航管理工具之OneNav
github:https://github.com/helloxz/onenav 解决痛点:经常使用的链接,时常被问起, 还要翻找之前的很老的记录,反反复复比较浪费平常的开发时间, 如果可以把这些常用 ...
- 简述GoLang优势与生态
开门见山,一睹golang的风采 性能优势 部署运维成本低 编码格式统一:测试简单 1. 性能优势 Go 语言被称为是:"21世纪的C语言",虽然这个帽子戴的有点高,不妨这里给大家 ...
- 0. RyuJIT Tutorials - RyuJIT 的历史和架构
目录 上一篇:无 下一篇:待更新 正文 RyuJIT - 即 .NET 的 JIT 编译器,负责将 IL 代码编译为最终用于执行的机器代码. 本系列为 RyuJIT 教程,将分为多篇进行更新发布,旨在 ...
- GooseFS 统一命名空间 | 加速存储业务访问
01 前言 GooseFS是 腾讯云存储团队推出的分布式缓存方案,主要针对需要缓存加速的数据湖业务场景,提供基于对象存储COS服务的近计算端数据加速层. 统一命名空间是GooseFS提供的透明命名 ...
- rabbitmq3.7.3 发布了一个新的 exchange x-random
direct exchange 同一个 routing key 可以绑定多个 queue,当给这个routing key发消息时,所有 queue 都会投递.这个行为对于一些场景不适用,有时我们希望只 ...