RSA加密是一种非对称加密,原理是:

  • 使⽤算法可以⽣成两把钥匙 A 和 B
  • 使⽤ A 加密的信息,使⽤ B 可以解开
  • 使⽤ B 加密的信息,使⽤ A 可以解开

⽇常使⽤中,我们把⼀把作为公钥公开发布。⼀把作为私钥,⾃⼰保留。这样,任何⼈都可以使⽤我们的公钥加密信息发给我们,我们则可以使⽤⾃⼰的私钥解开。

只要把私钥保存好,这个通信系统就⾮常安全。

数学基础

1. 欧拉函数

欧拉函数的输入是一个正整数,输出小于这个正整数的、跟它互质的整数数量。

定义是:

\[\phi(n)=n(1-\frac{1}{p_1})(1-\frac{1}{p_2})\cdots(1-\frac{1}{p_m})
\]

\(p_1, p_2, \cdots, p_m\) 表示 \(n\) 的 $ m $ 个质因子,重复的算一个。

比如5,它是个质数,只有它自己一个因子,所以\(\phi(5) = 5 \times (1 - \frac{1}{5}) = 4\)。

当然你也能猜出来,因为跟一个质数互质的数就是从1到它的前驱数的全部自然数,所以对于任意质数 \(p\) 都有

\[\phi(p) = p-1
\]

对于合数呢?比如 \(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)}\equiv 1 (\textbf{mod } 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\),使得

\[ab \equiv 1 (\textbf{mod } n)
\]

\(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是质数,所以就简单了

\[\phi(n) = (p -1)(q-1)
\]

在我们例子里就是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,使得

\[ax+by=d
\]

算法实现很简单:

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的过程:

\[m^{e} \equiv c (\textbf{ mod } n)
\]

假设我们要加密的消息是50,使用上面的计算

\[50^{13} \equiv c (\textbf{ mod } 77)
\]

可得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
}

解密过程

解密过程是一样的,依据是

\[c^{d} \equiv m (\textbf{ mod } n)
\]
    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)\) 呢?


\[\because m^{e} \equiv c (\textbf{ mod } n) \\
\therefore c= m^e-kn
\]

代入目标式:

\[({m^e-kn})^d\equiv m (\textbf{ mod } n)
\]

根据二项式定理,左边展开后除了第一项是$ m^{ed}$ 其余项都含有 \(kn\),必然是n的倍数,所以舍弃这些项。只要证明

\[m^{ed} \equiv m (\textbf{ mod } n)
\]

即可。

根据定义,

\[\because ed \equiv 1 (\textbf{ mod } z) \\
\therefore ed = 1+hz
\]

代入可得

\[m^{1+hz} = m^{h \phi(n) + 1} \equiv m (\textbf{ mod } n)
\]
  1. 当m和n互质时
\[\because m^{\phi(n)} \equiv 1(\textbf{ mod } 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了。

\[\because m^{\phi(q)} \equiv 1(\textbf{ mod } q) \\
\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)}
\]

又根据二项式定理

\[(kp)^{(q-1)h(p-1)} \equiv 1(\textbf{ mod } q) \\
\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:

\[(kp)^{ed} = kp + vpq \\
m^{ed} = m + vn \\
\therefore m^{ed} \equiv m (\textbf{ mod } n)
\]

得证。

RSA的原理和简单实践的更多相关文章

  1. Spring 学习二-----AOP的原理与简单实践

    一.Spring  AOP的原理 AOP全名Aspect-Oriented Programming,中文直译为面向切面(方面)编程.何为切面,就比如说我们系统中的权限管理,日志,事务等我们都可以将其看 ...

  2. 带你彻底理解RSA算法原理,很简单的

    1. 什么是RSA RSA算法是现今使用最广泛的公钥密码算法,也是号称地球上最安全的加密算法. 在了解RSA算法之前,先熟悉下几个术语 根据密钥的使用方法,可以将密码分为 对称密码 和 公钥密码 对称 ...

  3. RSA算法原理(简单易懂)

    1. 什么是RSA RSA算法是现今使用最广泛的公钥密码算法,也是号称地球上最安全的加密算法.在了解RSA算法之前,先熟悉下几个术语 根据密钥的使用方法,可以将密码分为对称密码和公钥密码 对称密码:加 ...

  4. kafka原理和实践(二)spring-kafka简单实践

    系列目录 kafka原理和实践(一)原理:10分钟入门 kafka原理和实践(二)spring-kafka简单实践 kafka原理和实践(三)spring-kafka生产者源码 kafka原理和实践( ...

  5. 轻松学习RSA加密算法原理

    转自:http://blog.csdn.net/sunmenggmail/article/details/11994013 http://blog.csdn.net/q376420785/articl ...

  6. 轻松学习RSA加密算法原理 (转)

    轻松学习RSA加密算法原理 (转) http://blog.csdn.net/q376420785/article/details/8557266 http://www.ruanyifeng.com/ ...

  7. Tengine HTTPS原理解析、实践与调试【转】

    本文邀请阿里云CDN HTTPS技术专家金九,分享Tengine的一些HTTPS实践经验.内容主要有四个方面:HTTPS趋势.HTTPS基础.HTTPS实践.HTTPS调试. 一.HTTPS趋势 这一 ...

  8. RSA加密算法原理及RES签名算法简介

    第一部分:RSA算法原理与加密解密 一.RSA加密过程简述 A和B进行加密通信时,B首先要生成一对密钥.一个是公钥,给A,B自己持有私钥.A使用B的公钥加密要加密发送的内容,然后B在通过自己的私钥解密 ...

  9. SSH原理与运用(一)和(二):远程登录 RSA算法原理(一)和(二)

    SSH原理与运用(一)和(二):远程登录  RSA算法原理(一)和(二) http://www.ruanyifeng.com/blog/2011/12/ssh_remote_login.html ht ...

  10. MapReduce 原理与 Python 实践

    MapReduce 原理与 Python 实践 1. MapReduce 原理 以下是个人在MongoDB和Redis实际应用中总结的Map-Reduce的理解 Hadoop 的 MapReduce ...

随机推荐

  1. 算法笔记——马拉核弹(Mana Nuclear)

    0x00 摘要 "马拉核弹"算法由 SXHT 同学(2009~今)发明,并在 2024 年 11 月于某不知名学校机房内正式公布.该算法基于 1975 年发明的 Manacher ...

  2. hyperf使用session

    在hyperf里面使用session的时候可以先安装组件包 composer require hyperf/session Session 组件的配置储存于  config/autoload/sess ...

  3. 2025年前端面试准备css篇

    1.css 盒子模型 css包含了内容(content) ,内边距(padding),边框(border),外边距(margin) 等因素. css 标准盒子模型宽包括:margin+border+p ...

  4. golang之性能分析工具pprof

    PProf 是一个 Go 程序性能分析工具,可以分析 CPU.内存等性能.Go 在语言层面上集成了 profile 采样工具,只需在代码中简单地引入 runtime/ppro 或者 net/http/ ...

  5. ClickHouse之物化MySQL

    Creates ClickHouse database with all the tables existing in MySQL, and all the data in those tables. ...

  6. Blazor 组件库 BootstrapBlazor 中Circle组件介绍

    组件介绍 Circle进度环组件,是一个图表类组件.一般有两种用途: 显示某项任务进度的百分比. 统计某些指标的占比. 它的样子如下: 它的代码如下: <Circle Width="2 ...

  7. Java Playwright 浏览器最大化

    Playwright 是一个用于自动化 Web 应用测试的现代工具,支持多种语言(包括 Java)及多个浏览器(如 Chromium.Firefox 和 WebKit).它提供了一致的 API 来控制 ...

  8. uniapp多次触发跳转问题

    问题描述:快速点击跳转页面后会闪退到登陆页面 解决方案:重新封装uniapp跳转api,加防抖锁,To.ts import { NavigateToOptions, RedirectToOptions ...

  9. 【自制FOC驱动器】深入浅出讲解FOC算法与SVPWM技术

    转载自 http://www.pengzhihui.xyz/2020/07/02/foc/ 前言前阵子工作太忙,好久没做个人项目了,最近久违地想做一个机器人项目,设计中需要用到高性能超小体积的伺服电机 ...

  10. 零售经营“新赛道” ——基于手机银行APP专区调研的客群精细化运营分析报告

    ​ 随着银行业竞争的不断深入及新客户增量日渐"到顶",各家银行的客群竞争逐渐由"跑马圈地"进入"精耕细作"的新阶段,在客群精准化服务方面不断 ...