序言
Fibonacci 数列
多项式时间算法:
var fib_arr = [0, 1];
function fib(n){
if(n == 0){
return 0;
}else if(n == 1){
return 1;
}
for (var i = 0; i < n-2; i++) {
fib_arr[n+2] = fib_arr[n+1] + fib_arr[n]
}
return fib_arr[n]
}
var start = new Date().getTime();
console.log(fib(30));
var end = new Date().getTime();
console.log('运行时间:'+(end-start));
// 运行时间:24
指数时间算法:
function fib(n){
if(n == 0){
return 0;
}else if(n == 1){
return 1;
}else{
return fib(n-1) + fib(n-2);
}
}
var start = new Date().getTime();
console.log(fib(30));
var end = new Date().getTime();
console.log('运行时间:'+(end-start));
// 运行时间:50
渐进
- 上界:Ο
- 下界:Ω
- 紧确界:Θ
数字的算法
基础算数
基数和对数
- b 为基数的 k 位数字最大为 b^k-1
- b 为基数表示 N 需要 ⌈log(b)(N+1)⌉ 位(应该是向上取整)
乘法(Al Khwarizmi 乘法规则)
类似使用二进制进行乘法
y(每次整除 2) | z(结果,由底至顶) |
---|---|
11 | 13+(13+(13*2)*2)*2 |
5 | 13+(13*2)*2 |
2 | 13*2 |
1 | 13 |
0 | 0 |
// 类似使用二进制进行乘法(当y>=0时可用)
function multiply(x, y){
if(0 == y){
return 0;
}
var z = multiply(x, Math.floor(y/2))
// y为偶数,y为奇数加到x上
return y%2 == 0 ? 2*z : x + 2*z;
}
console.log(multiply(13, 11));
除法
类似使用二进制进行除法
x(每次整除 2) | z(结果,由底至顶) |
---|---|
13 | [0+1, ((((0+1)*2+1)*2)*2+1)-11] |
6 | [0, ((0+1)*2+1)*2] |
3 | [0, (0+1)*2+1] |
1 | [0, (0+1)] |
0 | [0, 0] |
// 类似使用二进制进行除法(当y>=1时可用,q:quotient,r:remainder)
function divide(x, y){
if(0 == x){
return [0, 0];
}
var z = divide(Math.floor(x/2), y);
var q = 2 * z[0];
var r = 2 * z[1];
if (x%2 == 1) {
// 被除数为奇数除二余数为1
r++;
}
if (r >= y) {
// 余数大于除数,余数变为余数减去除数,商+1
r -= y;
q++;
}
console.log(q, r);
return [q, r];
}
console.log(divide(13, 11));
模运算
定义:x=qN+r
,且0<=r<N
,则 x 模 N 等于 r。
- x 与 y 模 N 同余 <=> x≡y(mod N) <=> N 整除 (x-y)
- 模运算准守结合律、分配率、交换律
模的指数运算
x:10, y:6, N:17
y(指数,每次整除 2) | z(结果,由底至顶) |
---|---|
6 | ((10*((10*1*1)%17)*((10*1*1)%17))%17)*((10*((10*1*1)%17)*((10*1*1)%17))%17)%17 |
3 | (10*((10*1*1)%17)*((10*1*1)%17))%17 |
1 | (10*1*1)%17 |
0 | 0 |
// 求模的指数运算(y>=1)
function modexp(x, y, N){
if(0 == y){
return 1;
}
var z = modexp(x, Math.floor(y/2), N);
if (y%2 == 0) {
return (z * z) % N;
}else{
return (x * z * z) % N;
}
}
console.log(modexp(11, 11, 2));//1
console.log(modexp(10, 11, 2));//0
console.log(modexp(10, 6, 17));//9
Euclid 最大公因数(greatest common divisor)算法
- Euclid 规则:如果 x,y 是正整数,且有 X>=y,那么 gcd(x, y) = gcd(x mod y, y)
// Euclid求最大公约数方法(a>=b>=0)
function euclid(a, b){
if(0 == b) return a;
return euclid(b, a % b);
}
console.log(euclid(11, 11));//11
console.log(euclid(33, 22));//11
console.log(euclid(97, 79));//1
- Euclid 拓展:用来求
gcd(a, b) = d = ax + by
的 x 和 y
// Euclid拓展(a>=b>=0)
function extend_euclid(a, b){
if(0 == b) return [1, 0, a];
var result = extend_euclid(b, a % b),
x = result[0],
y = result[1],
d = result[2];
return [y, x-Math.floor(a/b)*y, d];
}
console.log(extend_euclid(11, 11));//[ 0, 1, 11 ]
console.log(extend_euclid(33, 22));//[ 1, -1, 11 ]
console.log(extend_euclid(97, 79));//[ 22, -27, 1 ]
注:a mod b = a - ⌊a/b⌋*b
模的除法运算
不确定!!!
例,求 100/27 mod 97
求乘法逆元:
97*22 + 79*-27 = 1
=> 79*-27 = 1 mod 97
=> -79 = 1/27 mod 97
乘法逆元带入:
100/27 mod 97
= 100 * 1/27 mod 97
= 100 * -79 mod 97
= 54
素性测试
- 费马小定理:p 为素数,任意对于任意 1<=a<p 有
a^(p-1) = 1 mod p
// 素数测试算法(对于合数N,a的大多数无法通过测试)
function primality(N){
var a = Math.floor(2 + Math.random() * (N-3));
if (modexp(a, N - 1, N) == 1) {
return true;
}else{
return false;
}
}
// 求模的指数运算(y>=1),前面的算法
function modexp(x, y, N){
if(0 == y){
return 1;
}
var z = modexp(x, Math.floor(y/2), N);
if (y%2 == 0) {
return (z * z) % N;
}else{
return (x * z * z) % N;
}
}
console.log(primality(11027));
console.log(primality(11026));
// 素数测试算法(对于合数N,a的大多数无法通过测试)
function primality(N){
// 降低出错概率到(2^-100)
for (var i = 0; i < 100; i++) {
var a = Math.floor(2 + Math.random() * (N-3));
if (modexp(a, N - 1, N) != 1) {
return false;
}
}
return true;
}
// 求模的指数运算(y>=1),前面的算法
function modexp(x, y, N){
if(0 == y){
return 1;
}
var z = modexp(x, Math.floor(y/2), N);
if (y%2 == 0) {
return (z * z) % N;
}else{
return (x * z * z) % N;
}
}
console.log(primality(11027));
console.log(primality(11026));
密码学
一次一密乱码本(one-time pad)和 AES
使用密钥 r 进行异或加密解密
- 加密:er(x) = x ⊕ r
- 解密:er(er(x)) =(x ⊕ r) ⊕ r = x ⊕ (r ⊕ r) = x
一次一密乱码本
r 与发送信息长度一样
const r = "101011100";
function encrypt(x){
return (parseInt(x, 2) ^ parseInt(r, 2)).toString(2);
}
function unencrypt(x){
return (parseInt(x, 2) ^ parseInt(r, 2)).toString(2);
}
// test
var msg = "111110000";
var encrypted = encrypt(msg);
console.log(`encrypted: ${encrypted}`);
var unencrypted = unencrypt(encrypted);
console.log(`unencrypted: ${unencrypted}`);
AES
- r 通常定为 128 位(也有定为 192 位或 256 位的情况)
- 定义一种双向映射机制 er
- 信息分成片段,每个片段用 er 加密
RAS
生成密钥
- 找大整数 p,q(素数测试算法)
- 根据 N=pq,求 N(乘法)
- 找任意与 (p-1)(q-1) 互素的数 e
- 根据 ed≡(1 mod (p-1)(q-1)),求 d(拓展 Euclid 算法)
此时公钥为 (N,e),私钥为 (N,d)
class ras {
static genrsa() {
let p, q, N, e, d;
// 产生两个大素数
p = this._genpri(4);
q = this._genpri(4);
// 求N
N = p * q;
// 求e和d
[e, d] = this._get_eexp_and_inv((p - 1) * (q - 1));
return {
// 公钥
pub_key: {
N: N,
e: e
},
// 私钥
pri_key: {
N: N,
d: d
}
}
}
static encrypt(x, N, e) {
return this._modexp(x, e, N);
}
static unencrypt(y, N, d) {
return this._modexp(y, d, N);
}
// 生成素数
static _genpri(size) {
while (true) {
let n = Math.floor((1 << size) - 1 + Math.random() * (1 << size));
if (this._primality(n)) {
return n;
}
}
}
// 获取加密指数和乘法余元(encryption exponent and multiplicative inverse)
static _get_eexp_and_inv(N) {
while (true) {
let e = this._genpri(2);
let d = this._extend_euclid(N, e)[1];
// d > 1:乘法余元非负,且与N互素(模的指数运算无法使用负数)
if (d > 1) {
return [e, d];
}
}
}
// 模的指数运算
static _modexp(x, y, N) {
if (0 == y) {
return 1;
}
// static模式下arguments.callee失效
var z = this._modexp(x, Math.floor(y / 2), N);
if (y % 2 == 0) {
return (z * z) % N;
} else {
return (x * z * z) % N;
}
}
// 素数测试算法(对于合数N,a的大多数无法通过测试)
static _primality(N) {
// 降低出错概率到(2^-100)
for (var i = 0; i < 100; i++) {
var a = Math.floor(2 + Math.random() * (N - 3));
if (this._modexp(a, N - 1, N) != 1) {
return false;
}
}
return true;
}
// Euclid拓展(a>=b>=0)
static _extend_euclid(a, b) {
if (0 == b) return [1, 0, a];
var result = this._extend_euclid(b, a % b),
x = result[0],
y = result[1],
d = result[2];
return [y, x - Math.floor(a / b) * y, d];
}
}
// test
let msg = 123;
let test_res = ras.genrsa();
console.log(`test_res:`, test_res);
let encrypted = ras.encrypt(msg, test_res.pub_key.N, test_res.pub_key.e);
console.log(`encrypted: ${msg} -> ${encrypted}`);
let unencrypted = ras.unencrypt(encrypted, test_res.pri_key.N, test_res.pri_key.d);
console.log(`unencrypted: ${encrypted} -> ${unencrypted}`);
加密解密
- 加密:y=x^e mod N
- 解密:x=y^d mod N
破解?
- 方法 1:尝试所有可能的 x 判断
x^e = y mod N
是否成立 - 方法 2:对 N 因式分解,得到 p 和 q,进而计算出 d
散列表
散列表大小设为素数冲突概率小!
例:IP 地址的散列
// 未做解决冲突
function hash_IP(num) {
var a = [],
ip_table = new Array(num);
for (let i = 0; i < 4; i++) {
a[i] = Math.floor(Math.random() * num);
}
return function (IP) {
let d, sum = 0;
d = IP.split('.');
for (let i = 0; i < 4; i++) {
sum += +d[i] * a[i];
}
ip_table[Math.floor(sum % num)] = {
ip: IP,
data: IP
}
return ip_table;
}
}
var hi = hash_IP(100)
console.log(hi('127.0.0.1'));
console.log(hi('60.59.44.33'));
console.log(hi('192.110.119.115'));
分治算法
乘法
递归将乘数分为两段,并将 T(n) = 4T(n/2) + O(n)
的方法优化为
T(n) = 3T(n/2) + O(n)
的方法
效率:
由普通的 O(n^2)
提高到 O(3^log2(n))(约为 O(n^1.59))
PS: O(3^log2(n)) = O(n^log2(3))
指数对数一顿推就出来了
算法:
/**
* 分治整数乘法算法
* @param {number} x 被乘数,整数,大于等于0
* @param {number} y 乘数,整数,大于等于0
* @returns {number} 结果
*/
export default (x, y) => {
function multiply(x, y) {
var xl, xr, yl, yr, p1, p2, p3, n, nl, nr;
n = Math.max(x.toString(2).length, y.toString(2).length);
// 一位的时候x*y和x&y一样
if (n == 1) return x & y;
nl = Math.ceil(n / 2);
nr = Math.floor(n / 2);
xl = x >> nr;
xr = x - (xl << nr);
yl = y >> nr;
yr = y - (yl << nr);
p1 = multiply(xl, yl);
p2 = multiply(xr, yr);
p3 = multiply(xl + xr, yl + yr);
/**
* 左移的位数要在纸上写几遍把握一下(书上p1左移n位当n为偶数才能成立)
* 算数运算符优先级大于位移运算符
* 如:110 * 111 = ((11 << 1) + 0) * ((11 << 1) + 1)
* = [(11 * 11 << 2)] + [11 * 1 + 11 * 0 << 1] + [0 * 1]
* = [p1 << 2] + [((11 + 0) * (11 + 1) - p1 - p2) << 1] + [p2]
* = [p1 << 2] + [(p3 - p1 - p2) << 1] + [p2]
* 这个式子中n为3,nr为Math.floor(3 / 2)等于1
*/
return (p1 << (nr << 1)) + ((p3 - p1 - p2) << nr) + p2;
}
return multiply(x, y);
}
注:书中最后左移的位数只有偶数位才正确
主定理
T(n) = aT(n/b) + O(n^d)
- d > logb(a) :
T(n) = O(n^d)
- d = logb(a) :
T(n) = O((n^d)*log(n))
- d <logb(a) :
T(n) = O(n^(logb(a)))
递归树宽 a^logb(n),高 logb(n)
合并排序
T(n) = 2T(n/2) + O(n) = O(nlogn)
递归:
test_arr = [10, -1, 2, 3, 9, 10 ,9];
function mergesort(arr){
if (arr.length > 1) {
// 将两个排序好的数组进行连接
return merge(mergesort(arr.slice(0, Math.floor(arr.length/2))),
mergesort(arr.slice(Math.floor(arr.length/2), arr.length)));
}else{
return arr;
}
}
function merge(arr1, arr2){
if(arr1.length == 0) return arr2;
if(arr2.length == 0) return arr1;
if (arr1[0] > arr2[0]) {
return [arr1[0]].concat(merge(arr1.slice(1, arr1.length), arr2));
}else{
return [arr2[0]].concat(merge(arr2.slice(1, arr2.length), arr1));
}
}
console.log(mergesort(test_arr));
迭代:
test_arr = [10, -1, 2, 3, 9, 10 ,9];
function interative_mergesort(arr){
var temp_arr = test_arr.map((d)=>{
return [d];
});
while(temp_arr.length > 1){
temp_arr.push(merge(temp_arr.pop(), temp_arr.pop()))
}
return temp_arr[0];
}
function merge(arr1, arr2){
if(arr1.length == 0) return arr2;
if(arr2.length == 0) return arr1;
if (arr1[0] > arr2[0]) {
return [arr1[0]].concat(merge(arr1.slice(1, arr1.length), arr2));
}else{
return [arr2[0]].concat(merge(arr2.slice(1, arr1.length), arr1));
}
}
console.log(interative_mergesort(test_arr));
寻找中项(选择有序数组的某项)
用快速排序的思想,递归找出 SL(小于 v 的数)、SV(等于 v 的数)、 SR(大于 v 的数)缩小范围
效率:
最好:T(n) = T(n/2) + O(n) = O(n)
最坏:T(n) = T(n - 1) + O(n) = O(n^2)
PS:快速排序是超大文件排序算法的基础
矩阵乘法
使用T(n) = 7T(n/2) + O(n^2)
的分治算法可以从传统的 O(n^3)
优化到 O(n^log2(7))
留坑以后写
快速傅里叶变换(FFT)
预备知识
一个 d 次多项式被其在任意 d+1 个不同点出的取值所唯一确定
d+1 个不同点处的取值可以列出 d+1 个方程,解出系数 a0, a1, ..., ad(共 d+1 个)
多项式A(x)=a0*(x^0) + a1*(x^1) + ... + ad*(x^d)
的两种表示
- 系数:
a0, a1, ..., ad
- 值:
A(x0), A(x1), ..., A(xd)
系数与值的关系
系数 -> 计算 -> 值
值 -> 插值 -> 系数
问题转化
计算 A(x)*B(x) = C(x) 的问题转化成计算 A(x) 和 B(x) 在 2d+1(d 为多项式次数)个不同点处的取值的乘积,然后对这些乘积插值获取 C(x) 的系数
从直接计算和插值开始
下面有一个直接进行计算和插值求多项式乘积的算法,可以看出问题有:
- 计算:计算增广矩阵的时间复杂度是 O(n^2)(这里和直接乘起来一样复杂了)
- 插值:求非齐次线性方程组的时间复杂度是 O(n^3)(反而更加复杂了)
原本以为用 0,2,4,8... 这些点是不可以简化计算和插值,结果发现想多了简化不了,就用了 0,1,2,3... 这种取值。
FFT 正是巧妙地选择了计算的点使算法效率提高。
/**
* 使用未经优化的计算和插值方法,计算多项式A(x)*B(x)的系数
* @param {Array<number>} Ax 多项式A(x)的系数
* @param {Array<number>} Bx 多项式B(x)的系数
* @returns {Array<number>} 多项式A(x)*B(x)的系数
*/
export default (Ax, Bx) => {
let d,
am = [], //增广矩阵 augmented matrix
Cx = [];
if (Ax.length > Bx.length) {
d = Ax.length - 1;
Bx[Ax.length - 1] = 0;
while (Bx.push(0) != Ax.length);
} else if (Bx.length > Ax.length) {
d = Bx.length - 1;
while (Ax.push(0) != Bx.length);
} else {
d = Ax.length - 1;
}
// 计算增广矩阵(O(n^2))
for (let i = 0; i < 2 * d + 1; i++) { // 循环2*d + 1次
let x = 1,
Av = 0,
Bv = 0;
am[i] = [];
for (var j = 0; j < 2 * d + 1; j++) { // 循环2*d + 1次
if (j < d + 1) {
Av += Ax[j] * x;
Bv += Bx[j] * x;
}
am[i][j] = x;
x *= i;
}
am[i][j] = Av * Bv;
}
// 之后就是用增广矩阵求非齐次线性方程组了(O(n^3),比直接乘都复杂了)
// 1.三角阵
for (let i = 0; i < 2 * d; i++) {
for (let j = i + 1; j < 2 * d + 1; j++) {
let ratio = am[j][i] / am[i][i]
am[j] = am[j].map((v, index) => {
return v - ratio * am[i][index]
})
}
}
// 2.对角阵
for (let i = 2 * d; i > 0; i--) {
for (let j = i - 1; j > -1; j--) {
let ratio = am[j][i] / am[i][i]
am[j] = am[j].map((v, index) => {
return v - ratio * am[i][index]
})
}
}
// 3.单位阵(思想)
for (let i = 0; i < 2 * d + 1; i++) {
Cx[i] = Math.round(am[i][2 * d + 1] / am[i][i]);
}
return Cx;
}
开始 FFT
计算:点的选取
多项式A(x)
可以变为Ae(x^2) + x * Ao(x^2)
(也就是A(x) = Ae(x^2) + x * Ao(x^2)
),例如:
A(x) = 1*x^0 + 2*x^1 + 3*x^2 + 4*x^3 = (1*x^0 + 3*x^2) + x * (2*x^0 + 4*x^2)
这时如果选取的点为正负的数对进行计算,就可以简化将近一半的计算量,例如:
x1取1,x2取-1
这时计算:
A(x1) = Ae(x1^2) + x1 * Ao(x1^2)
A(x2) = Ae(x2^2) + x2 * Ao(x2^2)
只需计算:P1 = Ae(1) 和 P2 = 1 * Ao(1)
A(x1) = P1 + P2
A(x2) = P1 - P2
这里的 Ae(x) 和 Ao(x) 也是可以再分的,这样就可以使用分治了,但选取的_正负的数对的平方_还得是_正负的数对_这样才可以简化运算,这样的数存在么?当然存在那就是复数(这里得补点课),例如:
取4个点:
x1取1,x2取-1,x3取i,x4取-i
平方 1 -1
平方 1
取8个点:
取复数 1+0, -1+0, 0+i, 0-i, √2+√2i, -√2-√2i, √2-√2i, -√2+√2i
平方 1+0 -1+0 0+i 0-i
平方 1 -1
平方 1
以此类推,取16个点,取32个点。。。
插值:用计算的算法就可解决
用计算的算法就可解决插值问题,那么原理是什么呢?计算相当于n 阶的方阵 M * n 个系数
,插值相当于n 阶的方阵 M 的逆矩阵 * n 个计算出来的值
,且这个方阵满足M(w)^-1 = 1/n * M(w^-1)
,证明如下:
证明:略
图的分解
深度优先搜索(Depth-First-Search)
对图进行深度优先搜索得到森林
代码
Algorithms-JS/dfs.js at master · 1010543618/Algorithms-JS
有向图分解为强连通部件
步骤
- 找到出度为 0 的强连通部件(汇点强连通部件)
- 删去该强连通部件
- 重复执行 1 和 2,直到没有顶点
如何找到汇点强连通部件
因为对图 G 进行深度优先搜索得到的 post 最大的点(最后一个出栈的点)一定位于_源点强连通部件_中,
PS:注意对图进行深度优先搜索得到森林,而不是仅遍历可以达到的节点。这样从非源点强连通部件的节点开始遍历其 post 的值会加到下一个开始遍历的强连通部件中。例如:
(强连通部件C)--->---(强连通部件C')
若是从C开始遍历post最大的点(最后一个出栈的点)是C中的点
若是从C'开始遍历,post最大的点(最后一个出栈的点)还是C中的点
这样将图 G 翻转得到 GR,对 GR 进行深度优先搜索得到的 post 最大的点(最后一个出栈的点)一定位于_汇点强连通部件_中。
算法
- 将图 G 翻转得到 GR
- 对 GR 进行深度优先搜索,按出栈顺序(post 的值)降序排列顶点
- 依次从排序后的顶点开始进行深度优先搜索,找到汇点强连通部件
- 删除该汇点强连通部件
- 重复执行 3 和 4,直到没有顶点
代码
Algorithms-JS/get_strongly_connected_component.js at master · 1010543618/Algorithms-JS
图中的路径
广度优先搜索(Breadth-First Search)
广度优先搜索关注_某一顶点_的_所有能达到的顶点_的搜索深度。
贪心算法
最小生成树
Huffman 编码
Hron 公式
SAT 的一种特殊形式,可在线性时间求解。
Horn-satisfiability - Wikipedia
动态规划
最长递增子序列
编辑距离
问题:给字符串x[1...m]
与y[1...n]
,求使两个字符串对齐的最小代价(不同的列数)。
子问题:E(i,j)
,字符串x[1...m]
与y[1...n]
的前缀x[1...i]
与y[1...j]
的编辑距离。
子问题间关系:E(i,j) = min{1+E(i-1,j), 1+E(i,j-1), diff(i,j) + E(i-1,j-1)}
xy |
---|