古典密码是密码学发展的早期阶段,主要包括置换密码和代换密码两大类。 古典密码是指 1976 年以前的密码算法,主要通过手工或简单的机械操作进行信息的加密和解密。其核心思想是通过替换和置换来隐藏信息的真实内容。 通过将字母表中的每个字母移动固定位置来实现加密,是最简单的替换密码之一。 凯撒密码(英语:Caesar cipher),或称凯撒加密、凯撒变换、变换加密,是一种最简单且最广为人知的加密技术。凯撒密码是一种替换加密技术,明文中的所有字母都在字母表上向后(或向前)按照一个固定数目进行偏移后被替换成密文。例如,当偏移量是 3 的时候,所有的字母 A 将被替换成 D,B 变成 E,以此类推。这个加密方法是以罗马共和时期凯撒的名字命名的,据称当年凯撒曾用此方法与其将军们进行联系。 凯撒密码通常被作为其他更复杂的加密方法中的一个步骤,例如维吉尼亚密码。凯撒密码还在现代的 ROT13 系统中被应用。但是和所有的利用字母表进行替换的加密技术一样,凯撒密码非常容易被破解,而且在实际应用中也无法保证通信安全。 当偏移量是 3 的时候,所有的字母 A 将被替换成 D,B 变成 E,以此类推。 eg:
凯撒密码的替换方法是通过排列明文和密文字母表,密文字母表示通过将明文字母表向左或向右移动一个固定数目的位置。例如,当偏移量是左移 3 的时候(解密时的密钥就是 3): 使用时,加密者查找明文字母表中需要加密的消息中的每一个字母所在位置,并且写下密文字母表中对应的字母。需要解密的人则根据事先已知的密钥反过来操作,得到原来的明文。例如: 凯撒密码的加密、解密方法还能够通过同余的数学方法进行计算。首先将字母用数字代替,A=0,B=1,...,Z=25。此时偏移量为 n 的加密方法即为:
特定凯撒密码名称 根据偏移量的不同,还存在若干特定的凯撒密码名称: 即使使用唯密文攻击,凯撒密码也是一种非常容易破解的加密方式。可能有两种情况需要考虑: 攻击者知道(或者猜测)密码中使用了某个简单的替换加密方式,但是不确定是凯撒密码;
攻击者知道(或者猜测)使用了凯撒密码,但是不知道其偏移量。
对于第一种情况,攻击者可以通过使用诸如频率分析或者样式单词分析的方法,马上就能从分析结果中看出规律,得出加密者使用的是凯撒密码。
对于第二种情况,解决方法更加简单。由于使用凯撒密码进行加密的语言一般都是字母文字系统,因此密码中可能是使用的偏移量也是有限的,例如使用 26 个字母的英语,它的偏移量最多就是 25(偏移量 26 等同于偏移量 0,即明文;偏移量超过 26,等同于偏移量 1-25)。因此可以通过穷举法,很轻易地进行破解[6]。其中一种方法是在表格中写下密文中的某个小片段使用所有可能的偏移量解密后的内容——称为候选明文,然后分析表格中的候选明文是否具有实际含义,得出正确的偏移量,解密整个密文。例如,被选择出的密文片段是"EXXEGOEXSRGI",可以从右表中的候选明文里很快看出其正确的偏移量是 4。也可以通过在每一个密文单词的每一个字母下面,纵向写下整个字母表其他字母,然后可以通过分析,得出其中的某一行便是明文。 另外一种攻击方法是通过频率分析。当密文长度足够大的情况下,可以先分析密文中每个字母出现的频率,然后将这一频率与正常情况下的该语言字母表中所有字母的出现频率做比较。例如在英语中,正常明文中字母 E 和 T 出现的频率特别高,而字母 Q 和 Z 出现的频率特别低,而在法语中出现频率最高的字母是 E,最低的是 K 和 W。可以通过这一特点,分析密文字母出现的频率,可以估计出正确的偏移量。此外,有时还可以将频率分析从字母推广到单词,例如英语中,出现频率最高的单词是[7]:the, of, and, a, to, in...。通过将最常见的单词的所有可能的 25 组密文,编组成字典,进行分析。比如 QEB 可能是 the,MPQY 可能是单词 know(当然也可能是 aden)。但是频率分析也有其局限性,它对于较短或故意省略元音字母或者其他缩写方式写成的明文加密出来的密文进行解密并不适用。 另外,通过多次使用凯撒密码来加密并不能获得更大的安全性,因为使用偏移量 A 加密得到的结果再用偏移量 B 加密,等同于使用 A+B 的偏移量进行加密的结果。
用典型的英语书写的文字样本中各字母出现频率
用典型的法语书写的文字样本中各字母出现频率 使用一个关键词来对明文进行多表替换加密,是古典密码中较为复杂的一种。 维吉尼亚密码(法语:Chiffre de Vigenère,又译维热纳尔密码)是使用一系列凯撒密码组成密码字母表的加密算法,属于多表密码的一种简单形式。 维吉尼亚密码曾多次被发明。该方法最早记录在吉奥万·巴蒂斯塔·贝拉索( Giovan Battista Bellaso)于 1553 年所著的书《吉奥万·巴蒂斯塔·贝拉索先生的密码》(意大利语:La cifra del. Sig. Giovan Battista Bellaso)中。然而,后来在 19 世纪时被误传为是法国外交官布莱斯·德·维吉尼亚(Blaise De Vigenère)所创造,因此现在被称为“维吉尼亚密码”。 维吉尼亚密码以其简单易用而著称,同时初学者通常难以破解,因而又被称为“不可破译的密码”(法语:le chiffre indéchiffrable)。这也让很多人使用维吉尼亚密码来加密的目的就是为了将其破解。
在一个凯撒密码中,字母表中的每一字母都会作一定的偏移,例如偏移量为 3 时,A 就转换为了 D、B 转换为了 E……而维吉尼亚密码则是由一些偏移量不同的凯撒密码组成。 为了生成密码,需要使用表格法。这一表格包括了 26 行字母表,每一行都由前一行向左偏移一位得到。具体使用哪一行字母表进行编译是基于密钥进行的,在过程中会不断地变换。 例如,假设明文为: 选择某一关键词并重复而得到密钥,如关键词为 LEMON 时,密钥为: 对于明文的第一个字母 A,对应密钥的第一个字母 L,于是使用表格中 L 行字母表进行加密,得到密文第一个字母 L。类似地,明文第二个字母为 T,在表格中使用对应的 E 行进行加密,得到密文第二个字母 X。以此类推,可以得到:
用来加密解密的维吉尼亚表格 解密的过程则与加密相反。 eg:
根据密钥第一个字母 L 所对应的 L 行字母表,发现密文第一个字母 L 位于 A 列,因而明文第一个字母为 A。密钥第二个字母 E 对应 E 行字母表,而密文第二个字母 X 位于此行 T 列,因而明文第二个字母为 T。以此类推便可得到明文。 用数字 0-25 代替字母 A-Z,维吉尼亚密码的加密文法可以写成同余的形式: 对包括维吉尼亚密码在内的所有多表密码的破译都是以字母频率为基础的,但直接的频率分析却并不适用。例如,如果 P 是密文中出现次数最多的字母,则 P 很有可能对应 E(前提是明文的语言为英语)。原因在于 E 是英语中使用频率最高的字母。然而,由于在维吉尼亚密码中,E 可以被加密成不同的密文,因而简单的频率分析在这里并没有用。 破译维吉尼亚密码的关键在于它的密钥是循环重复的。如果我们知道了密钥的长度,那密文就可以被看作是交织在一起的凯撒密码,而其中每一个都可以单独破解。使用卡西斯基试验和弗里德曼试验来得到密钥的长度。 卡西斯基试验
弗里德里希·卡西斯基于 1863 年首先发表了完整的维吉尼亚密码的破译方法,称为卡西斯基试验(Kasiski examination)。早先的一些破译都是基于对于明文的认识、或者使用可识别的词语作为密钥。而卡西斯基的方法则没有这些限制。然而,在此之前,已经有人意识到了这一方法。1854 年,英国数学家、发明家兼机械工程师查尔斯·巴贝奇受到斯维提斯(John Hall Brock Thwaites)在《艺术协会杂志》(Journal of the Society of the Arts)上声称发明了“新密码”的激励,从而破译了维吉尼亚密码。巴贝奇发现斯维提斯的密码只不过是维吉尼亚密码的一个变种而已,而斯维提斯则向其挑战,让他尝试破译用两个不同长度的密钥加密的密文。巴贝奇成功地进行了破译,得到的明文是丁尼生所写的诗《罪恶的想象》(The Vision of Sin),使用的密钥则是丁尼生妻子的名字 Emily(艾米莉)。巴贝奇从未对他的方法进行过解释 。在对巴贝奇生前笔记的研究中发现,早在 1846 年巴贝奇就使用了这一方法,与后来卡西斯基发表的方法相同。 卡西斯基试验是基于类似 the 这样的常用单词有可能被同样的密钥字母进行加密,从而在密文中重复出现。例如,明文中不同的 CRYPTO 可能被密钥 ABCDEF 加密成不同的密文: 此时明文中重复的元素在密文中并不重复。然而,如果密钥相同的话,结果可能便为(使用密钥 ABCD): 此时卡西斯基试验就能产生效果。对于更长的段落此方法更为有效,因为通常密文中重复的片段会更多。如通过下面的密文就能破译出密钥的长度: 其中,两个 DYDUXRMH 的出现相隔了 18 个字母。因此,可以假定密钥的长度是 18 的约数,即长度为 18、9、6、3 或 2。而两个 NQD 则相距 20 个字母,意味着密钥长度应为 20、10、5、4 或 2。取两者的交集,则可以基本确定密钥长度为 2。 弗里德曼试验
弗里德曼试验由威廉·F·弗里德曼(William F. Friedman)于 1920 年代发明。他使用了重合指数(index of coincidence)来描述密文字母频率的不匀性,从而破译密码。
κp 指目标语言中两个任意字母相同的概率(英文中为 0.067),
κr 指字母表中这种情况出现的概率(英文中为 1/26=0.0385),从而密钥长度可以估计为:
其中,c 是指字母表的长度(英文为 26),N 指文本的长度,n1 到 nc 是指密文的字母频率,为整数。 此方法只是一种估计,会随着文本长度的增加而更为精确。在实践中,会尝试接近此估计的多个密钥长度。一种更好的方法是将密文写成矩阵形式,其中列数与假定的密钥长度一致,将每一列的重合指数单独计算,并求得平均重合指数。对于所有可能的密钥长度,平均重合指数最高的最有可能是真正的密钥长度。这样的试验可以作为卡西斯基试验的补充。 频率分析:
一旦能够确定密钥的长度,密文就能重新写成多列,列数与密钥长度对应。这样每一列其实就是一个凯撒密码,而此密码的密钥(偏移量)则对应于维吉尼亚密码密钥的相应字母。与破译凯撒密码类似的方法,就能将密文破译。 柯克霍夫方法作为卡西斯基试验的改进,由奥古斯特·柯克霍夫(Auguste Kerckhoffs)提出。它将每一列的字母频率与转换后的明文频率相对应而得出每一列的密钥字母。一旦密钥中每一个字母都能确定,就能很简单地破译密文,从而得到明文。如果维吉尼亚字母表表格本身是杂乱而非按通常字母表顺序的话,那柯克霍夫方法就会无效,但卡西斯基试验和重复指数对于决定密钥长度仍旧是有效的。 将明文按列写入,密文按行输出,通过置换字母的位置来加密。 栅栏密码是古典密码的一种,其原理是将一组要加密的明文划分为 n 个一组(n 通常根据加密需求确定,且一般不会太大,以保证密码的复杂性和安全性),然后取每个组的第一个字符(有时也涉及取其他位置的字符,但规则需事先约定),根据情况将这些字符重新排列组合成一段无规律的话,形成密文。 栅栏密码的“栏”数,即分组后形成的“列”数或“行”数(具体取决于加密时的排列方式),是栅栏密码的一个重要参数。根据栏数的不同,栅栏密码可以分为多种类型,其中比较常见的是 2 栏栅栏密码。但理论上,栏数 n 可以是任何正整数(实际应用中受限于明文的长度和加密的安全性要求)。 分栏:首先将需要加密的明文(即原始信息)中的字母交替排成多行(通常称为“栏”),栏数可以是任意的,但通常不会超过 30 个字母(大约一两句话的长度) 重新组合:接着,按照特定的顺序(例如,从上到下,再从下到上)将各栏中的字母重新组合成一行,形成密文。 以明文“THE LONGEST DAY MUST HAVE AN END”为例,如果选择两栏进行加密,过程如下: 重新组合:将第二栏的字母放在第一栏的后面,得到密文 splitString 函数: 主程序: 解密是加密的逆过程。已知密文和栏数,解密者可以按照相反的顺序将密文重新排列成多栏,然后再将各栏的字母按照原始的顺序合并成明文。 例如,对于上述密文“TEOGSDYUTAENN HLNETAMSHVAED”,已知为两栏加密,解密过程如下: 重新组合:按照上下上下的顺序将两行字母合并成一行,得到明文“THE LONGEST DAY MUST HAVE AN END”。 结合了凯撒密码和乘法密码的特点,通过线性变换进行加密。 仿射密码具有可逆性的条件是:
设密钥 K= (7, 3), 用仿射密码加密明文 hot。 三个字母对应的数值是 7、14 和 19。 分别加密如下: 三个密文数值为 0、23 和 6,对应的密文是 AXG。 先来引入一个定义.
大家知道, 好多东西都有逆, 大家读小学时都知道,两个数相乘乘积为 1,则互为倒数, 其实是最简单的逆.
后来, 我们到了高中, 我们学习了逆函数;
到了大学, 我们学习线性代数,知道两个矩阵的乘积为单位矩阵的话,则这两个矩阵互为逆矩阵.
现在我跟大家介绍另一种逆. 叫模逆. 其实很好理解的,
如下:
若 a,b 两数的乘积对正整数 n 取模的结果为 1. 则称 a,b 互为另外一个的模逆.
比如: 37 = 21; 21 % 20 = 1 ; 所以 3,7 互为 20 的 模逆.
93 = 27; 27 % 26 = 1 ; 所以 9,3 互为 26 的 模逆.
如何标记? 利用 java 代码求模逆 利用 c++实现加解密 通过点和划的不同组合来表示不同的字母和数字,主要用于电报通信。 摩尔斯电码(Morse code)也被称作摩斯密码,是一种时通时断的信号代码,通过不同的排列顺序来表达不同的英文字母、数字和标点符号。它发明于 1837 年,是一种早期的数字化通信形式。不同于现代化的数字通讯,摩尔斯电码只使用零和一两种状态的二进制代码,它的代码包括五种:短促的点信号“・”,保持一定时间的长信号“—”,表示点和划之间的停顿、每个词之间中等的停顿,以及句子之间长的停顿。 摩尔斯电码的一个特点就是所有的字母并不是等长的,我们可以从它的编码表中直观的看到。
其实换一种方式来看这张编码表,会更好理解,因为这其实就是一棵霍夫曼树(Huffman Tree)。
首先,我们问一个问题:摩尔斯电码,只包含点和中横线吗? 如果不仔细翻看资料,可能大部分人会认为摩尔斯电码是由 · 和 - 组成的,也就是我们常说的「点」和「中横线」,也可以写作 但是实际上,无论从什么角度来考虑,摩尔斯电码都不止包含这两个元素。 先来简单测试一下,如果我们只使用 那么,假如我发送了 ... ,我要发送的是 S 呢?还是 EEE 呢?还是 T 呢?
难道分隔符被隐藏了? 此时我们似乎发现了一个隐藏的字符,那就是空格,或者叫分隔符。 我们都知道,使用摩尔斯电码发送电报的时候需要使用「电键」,而电键的基础原理其实就是控制电路在「通」「断」两种状态间切换,通过按下时长的长短来区分「点」「中横线」。
也就是说,「点」与「点」之间,「字符」与「字符」之前,其实是通过间隔来进行分隔的。 我们假定一个「点」的时长为一个单位长度 T,那么「中横线」的时长就是 3T。 同一个字符内部的「点」「中横线」间的间隔也是 T,字符间的间隔是 3T,而单词间的间隔则是 7T。 此处我们选择两个字符 O 和 R 作为示意,在发报的过程中,电路的通断情况其实是下面这样的:
这样,我们就可以用简单的「通」「断」来表示出摩尔斯电码了。 其实看到上面的示意图就会发现,其实电路状态的「通」「断」,完全可以理解为二进制的 1 和 0,而摩尔斯电码的基本组成结构,也可以理解为只有「通」「断」。 当然,我们也可以理解按常见的方式去理解它的结构,只是这下我们知道它背后的底层内容了。 其实换一个角度也很容易想明白,基于电路通断的摩尔斯电码,不论上层怎么定义,都不可能绕开底层只有两种状态这件事。 从上面的示意图中,其实我们还能发现另一个问题,摩尔斯电码的效率并不高。 发送 OR 两个字符,总共用了 21T 的时长,操作手按下电键 7 次,这个效率用作信息传递不免让人觉得有些焦躁。 于是,就有了各种提升摩尔斯电码效率的操作。 需要说明的是,需要被压缩的并不是摩尔斯电码,而是英文单词。 正如许多人使用 tks 来代指 thanks ,BTW 来代指 by the way 一样,英文世界里也存在着大量的对常用词汇的缩写。一百多年前的前辈们,也是对它进行了充分的利用。 这些缩写分为两类,一类是简单的拼接字符,形成短词,相当于单词缩写。 另一类则是直接造了新的字符出来,称为特殊符号(统一符号)。 通过将字母写在缠绕在圆木棍上的羊皮纸上,然后重新展开来读取加密信息。 在密码学里,密码棒是个可使的传递讯息字母顺序改变的工具,由一条加工过、且有夹带讯息的皮革绕在一个木棒所组成。在古希腊,文书记载着斯巴达人用此于军事上的讯息传递。 密码接受者需使用一个相同尺寸、让他将密码条绕在上面解读的棒子。快速且不容易解读错误的优点,使它在战场上大受欢迎。 但是它很容易就被破解了。 假设那棒可写下四个字母使之围绕成圆圈且 5 个字母可连成一线。 解密的方法其实就是将编码方法反过来: 使用一个 5x5 的方阵来加密字母,属于多表代换密码。
例: 在这个方阵中,每个字母都被分配了两个数字,第一个数字表示行号,第二个数字表示列号。例如: 以此类推,直到 Z 对应 55。 接收密文:接收到密文后,需要将其转换回原始的字母。 数字转换:根据方阵,将每个数字对转换回对应的字母。例如,如果收到的密文是 "13 22 12 12 15",则转换回 "HELLO"。 特点和局限性 简单性:波利比奥斯方阵的加密和解密过程相对简单,易于理解和实现。 可视化:方阵提供了一种直观的方式来查看和记忆字母与数字的对应关系。 频率分析:尽管将字母转换为数字,但字母的频率模式仍然存在。如果攻击者知道语言的常用字母频率,他们可以对密文进行频率分析,尝试破解密码。 扩展性:波利比奥斯方阵可以扩展到更大的方阵,以包含更多的字符,例如小写字母、数字或特殊字符。 安全性:在现代密码学标准下,波利比奥斯方阵被认为是不安全的,因为它容易受到各种密码分析技术的攻击。 使用密钥控制生成矩阵,然后每两个字母为单位进行代换。 Playfair 密码是一种使用一个关键词方格来加密字符对的加密算法,是一种多表代换的对称加密技术。 它依据一个 5*5 的正方形组成的密码表来编写,密码表里排列有 25 个字母。如果一种语言字母超过 25 个,可以去掉使用频率最少的一个。如,法语一般去掉 w 或 k,德语则是把 i 和 j 合起来当成一个字母看待。英语中 z 使用最少,可以去掉它。可以很好的防御频率分析法的攻击。 同时 Playfair 密码对明文和密文有一定的要求:在加密之前首先需要整理明文,将明文中两个字母组成一组,如果两个字母相同则分成两组,在每组的后面加字母 X 或 Q。如果明文字母个数是奇数,在最后一个字母之后加字母 X 或 Q。密文的字母个数一定是偶数,任意两个同组的字母都不会相同。 若 m1,m2 在同一行,对应密文 c1,c2 分别是紧靠 m1,m2 右端的字母。其中第一列被看做是最后一列的右方。 若 m1,m2 在同一列,对应密文 c1,c2 分别是紧靠 m1,m2 下方的字母。其中第一行被看做是最后一行的下方。 若 m1,m2 不在同一行,不在同一列,则 c1,c2 是由 m1,m2 确定的矩形的其他两角的字母(横向替换或者纵向替换)。 若 c1,c2 在同一行,对应明文 m1,m2 分别是紧靠 c1,c2 左端的字母。其中第一列被看做是最后一列的右方。 若 c1,c2 在同一列,对应明文 m1,m2 分别是紧靠 c1,c2 上方的字母。其中第一行被看做是最后一行的下方。 若 c1,c2 不在同一行,不在同一列,则 m1,m2 是由 c1,c2 确定的矩形的其他两角的字母(横向替换或者纵向替换)。 加密时,明文中的每个字母都会转换成 5 个为一组由 a 和 b 组成的英文字母。 培根密码,也称为倍康尼密码,是由英格兰哲学家、政治家、作家、科学家弗朗西斯·培根发明的一种隐写术其原理是通过使用两种不同长度的序列来代表不同的字母,从而将加密信息隐藏在普通文本中。 1、第一种方式: 2、第二种方式: 一种以格子为基础的简单替代密码,一一对应的。 猪圈密码(Pigpen cipher,亦称朱高密码、共济会密码)是一种以格子为基础的简单替代式密码。即使使用符号,也不会影响密码分析,亦可用在其它替代式的方法。 猪圈密码由于其图形化的特点,在早期被一些秘密组织(如共济会传闻中使用)用于保密通信,但随着密码学的发展,它已经被现代更强大的加密技术所取代,不过它仍然是密码学历史上一个有趣的研究对象。 古典密码主要应用于军事和外交领域,用于保护机密信息。例如,凯撒密码曾被罗马帝国的恺撒大帝用于战争通信。 尽管古典密码在历史上发挥了重要作用,但其安全性较低,容易被破解。例如,凯撒密码的密钥空间较小,容易受到频率分析等攻击方法的破解。古典密码
古典密码的概念
古典密码的定义
古典密码的分类
古典密码的常见算法
凯撒密码
明文字母表:ABCDEFGHIJKLMNOPQRSTUVWXYZ
密文字母表:DEFGHIJKLMNOPQRSTUVWXYZABC明文:THE QUICK BROWN FOX JUMPS OVER THE LAZY DOG
密文:WKH TXLFN EURZQ IRA MXPSV RYHU WKH ODCB GRJ
密码破解
偏移量
候选明文
0
exxegoexsrgi
1
dwwdfndwrqfh
2
cvvcemcvqpeg
3
buubdlbupodf
4
attackatonce
5
zsszbjzsnmbd
6
yrryaiyrmlac
……………………………
………………………………
23
haahjrhavujl
24
gzzgiqgzutik
25
fyyfhpfytshj
凯撒在线解密网站:https://ctf.bugku.com/tool/caesar
维吉尼亚密码
ATTACKATDAWN
LEMONLEMONLE
明文:ATTACKATDAWN
密钥:LEMONLEMONLE
密文:LXFOPVEFRNHR密码破解
密钥:ABCDEF AB CDEFA BCD EFABCDEFABCD
明文:CRYPTO IS SHORT FOR CRYPTOGRAPHY
密文:CSASXT IT UKSWT GQU GWYQVRKWAQJB密钥:ABCDAB CD ABCDA BCD ABCDABCDABCD
明文:CRYPTO IS SHORT FOR CRYPTOGRAPHY
密文:CSASTP KV SIQUT GQU CSASTPIUAQJB密文:DYDUXRMHTVDVNQDQNWDYDUXRMHARTJGWNQD
更细致的破解讲解可以参考:https://blog.csdn.net/qq_42650988/article/details/104343051
维吉尼亚密码的变体
栅栏密码
加密原理
分栏:
第一栏:T E O G S D Y U T A E N N
第二栏:H L N E T A M S H V A E DTEOGSDYUTAENN HLNETAMSHVAED
加密算法
# Rail-Fence Cipher Encode Algorithm
"""
加密信息 THE LONGEST DAY MUST HAVE AN END
key = 2
分栏第一栏:T E O G S D Y U T A E N N
分栏第二栏:H L N E T A M S H V A E D
将第二栏的字母放在第一栏的后面,得到密文 TEOGSDYUTAENNHLNETAMSHVAED
"""
def splitString(string, step):
length = len(string)
count = length % step
if length % step != 0:
for i in range(step - count):
string += ' '
# 将字符串分隔成若干个子串, 子串最大长度为 key
return [string[i: i + step] for i in range(0, length, step)]
if __name__ == '__main__':
string = input("Encode Info:").replace(' ', '')
key = int(input("Key:"))
secret = ''
tempList = splitString(string, key)
# 遍历二维数组, 取出每栏中的字符串
for i in range(key):
for j in range(len(tempList)):
secret += tempList[j][i]
print("Encode-Secret: %s" % secret)代码说明
splitString
函数将字符串分割成二维列表。使用示例
Encode Info: THE LONGEST DAY MUST HAVE AN END
Key: 2
Encode-Secret: TEOGSDYUTAENNHLNETAMSHVAED解密原理
分栏:将密文分成两行
第一栏:T E O G S D Y U T A E N N
第二栏:H L N E T A M S H V A E D解密算法
# Rail-Fence Cipher Decode
"""
栅栏密码解密(即栅栏密码加密逆过程)
解密思路如下:
1. 分栏计算长度等于最大长度的栏的个数和长度等于最小长度的栏的个数
2. 分栏之后执行加密的逆过程
"""
def split_string(string, step):
length = len(string)
if length % step == 0:
# 计算子串长度
child_string_length = int(length / step)
return child_string_length, [string[i:i + child_string_length] for i in range(0, length, child_string_length)]
else:
# 最大长度和最小长度之差恒等于1
min_string_length = int(length / step)
max_string_length = min_string_length + 1
max_length_count = length % step
min_length_count = step - max_length_count
max_string_length_sum = (min_string_length + 1) * max_length_count
# 将长度等于最大长度的栏和长度等于最小长度的栏分离
max_string_sum = string[0:max_string_length_sum]
min_string_sum = string[max_string_length_sum:]
# 将字符串彻底分栏
max_string_list = [max_string_sum[i:i + max_string_length] for i in range(max_length_count)]
# 注意: 在分离最小长度栏时需要在每个子串后面加上空字符 ' ', 防止后面索引越界
min_string_list = [min_string_sum[i * min_string_length:i * min_string_length + min_string_length] + ' ' for i in range(min_length_count)]
# 将最大长度栏集合和最小长度栏集合进行合并
string_list = max_string_list + min_string_list
return max_string_length, string_list
if __name__ == '__main__':
encode_secret = input("encodeSecret: ")
key = int(input("key: "))
child_string_length, temp_list = split_string(encode_secret, key)
secret = ''
for i in range(child_string_length):
for j in range(key):
secret += temp_list[j][i]
secret_info = secret.replace(' ', '')
print(secret_info)主要改动点:
str()
转换,因为 input()
函数已经返回字符串类型。仿射密码
加解密函数
密钥:k = (m, n)
C = Ek(m) = (k1*m + k2) mod n ;
M = Dk(c) = k3(c-k2) mod n (其中(k3×k1) mod 26 = 1);gcd(k1, n)=1. 当 k1=1
时,仿射密码变为加法密码,当 k2=0
时,仿射密码变为乘法密码。
仿射密码中的密钥空间的大小为nφ(n)
,当 n 为 26 字母,φ(n)=12
,因此仿射密码的密钥空间为12×26 = 312
。加密
(7×7 + 3) mod 26 = 52 mod 26 =0
(7×14 + 3) mod 26 = 101 mod 26 =23
(7×19 + 3) mod 26 =136 mod 26 =6解密
import java.util.Scanner;
/**
* 仿射密码的加密和解密
* n = 26
*/
public class AffineCipher {
// 模数
private static final int N = 26;
public static void main(String[] args) {
try (Scanner scanner = new Scanner(System.in)) {
// 输入密钥
int k1 = readKey(scanner, "请输入密钥k1 (与26互质的整数): ");
int k2 = readKey(scanner, "请输入密钥k2 (0-25之间的整数): ");
// 输入明文
System.out.println("请输入明文:");
String plaintext = scanner.nextLine();
// 加密
String ciphertext = encrypt(k1, k2, plaintext);
System.out.println("密文: " + ciphertext);
// 解密
String decryptedText = decrypt(k1, k2, ciphertext);
System.out.println("解密结果: " + decryptedText);
}
}
/**
* 读取并验证密钥
*/
private static int readKey(Scanner scanner, String prompt) {
int key;
while (true) {
System.out.print(prompt);
if (scanner.hasNextInt()) {
key = scanner.nextInt();
if (prompt.contains("k1") && gcd(key, N) != 1) {
System.out.println("错误: k1 必须与 26 互质。请重新输入。");
} else if (prompt.contains("k2") && (key < 0 || key >= N)) {
System.out.println("错误: k2 必须在 0 到 25 之间。请重新输入。");
} else {
break;
}
} else {
System.out.println("错误: 请输入一个整数。");
scanner.next(); // 清除无效输入
}
}
return key;
}
/**
* 计算最大公约数
*/
private static int gcd(int a, int b) {
while (b != 0) {
int temp = a % b;
a = b;
b = temp;
}
return a;
}
/**
* 加密操作
*/
private static String encrypt(int k1, int k2, String plaintext) {
StringBuilder ciphertext = new StringBuilder();
for (char c : plaintext.toCharArray()) {
if (Character.isUpperCase(c)) {
ciphertext.append((char) ((k1 * (c - 'A') + k2) % N + 'A'));
} else if (Character.isLowerCase(c)) {
ciphertext.append((char) ((k1 * (c - 'a') + k2) % N + 'a'));
} else {
ciphertext.append(c); // 保留非字母字符
}
}
return ciphertext.toString();
}
/**
* 解密操作
*/
private static String decrypt(int k1, int k2, String ciphertext) {
int k3 = modInverse(k1, N);
StringBuilder plaintext = new StringBuilder();
for (char c : ciphertext.toCharArray()) {
if (Character.isUpperCase(c)) {
plaintext.append((char) ((k3 * (c - 'A' - k2 + N)) % N + 'A'));
} else if (Character.isLowerCase(c)) {
plaintext.append((char) ((k3 * (c - 'a' - k2 + N)) % N + 'a'));
} else {
plaintext.append(c); // 保留非字母字符
}
}
return plaintext.toString();
}
/**
* 计算模逆元
*/
private static int modInverse(int a, int m) {
a = a % m;
for (int x = 1; x < m; x++) {
if ((a * x) % m == 1) {
return x;
}
}
throw new ArithmeticException("No modular inverse found!");
}
}#include <iostream>
#include <fstream>
#include <string>
#include <vector>
class AffineCipher {
private:
std::string fileStr;
std::string finalStr;
int a, b;
std::vector<int> itable = {0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25};
std::vector<char> Ctable = {'A','B','C','D','E','F','G','H','I','J','K','L','M','N','O','P','Q','R','S','T','U','V','W','X','Y','Z'};
std::vector<char> ctable = {'a','b','c','d','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','t','u','v','w','x','y','z'};
int gcd(int a, int b) {
return b == 0 ? a : gcd(b, a % b);
}
int getInv(int a, int m) {
int x, y;
gcd(a, m, x, y);
return (x % m + m) % m;
}
public:
void init() {
fileStr.clear();
finalStr.clear();
a = 0;
b = 0;
}
void showMenu() {
std::cout << "*****************仿射密码*****************" << std::endl;
std::cout << "\t\t1.加密文件" << std::endl;
std::cout << "\t\t2.解密文件" << std::endl;
std::cout << "\t\t3.退出" << std::endl;
std::cout << "******************************************" << std::endl;
}
void keyDown() {
int userkey;
std::cin >> userkey;
switch (userkey) {
case 1:
encryptFile();
break;
case 2:
decryptFile();
break;
case 3:
exit(0);
break;
default:
std::cout << "无效的选择,请重新输入。" << std::endl;
}
}
void readFile() {
std::string fileName;
std::cout << "请输入文件名:" << std::endl;
std::cin >> fileName;
std::ifstream infile(fileName);
if (!infile) {
std::cout << "未找到相关文件" << std::endl;
return;
}
fileStr = std::string((std::istreambuf_iterator<char>(infile)), std::istreambuf_iterator<char>());
infile.close();
std::cout << "成功打开文件" << std::endl;
std::cout << "待处理的文件为:" << std::endl;
std::cout << fileStr << std::endl;
}
void saveFile() {
std::string fileName;
std::cout << "请输入要保存信息的文件名:" << std::endl;
std::cin >> fileName;
std::ofstream outfile(fileName);
if (!outfile) {
std::cout << "保存文件失败" << std::endl;
return;
}
outfile << finalStr;
outfile.close();
std::cout << "保存成功" << std::endl;
}
void encryptFile() {
readFile();
std::cout << "请输入密钥(a,b):" << std::endl;
std::cin >> a >> b;
finalStr.clear();
for (char ch : fileStr) {
if (ch == ' ') {
finalStr += ' ';
} else {
int temp = (a * itable[ch - 'a'] + b) % 26;
finalStr += Ctable[temp];
}
}
std::cout << "得到的密文为:" << std::endl;
std::cout << finalStr << std::endl;
saveFile();
}
void decryptFile() {
readFile();
std::cout << "请输入密钥(a,b):" << std::endl;
std::cin >> a >> b;
int ainv = getInv(a, 26);
finalStr.clear();
for (char ch : fileStr) {
if (ch == ' ') {
finalStr += ' ';
continue;
}
int temp = ainv * (itable[ch - 'A'] - b) % 26;
temp = (temp % 26 + 26) % 26;
finalStr += ctable[temp];
}
std::cout << "得到的明文为:" << std::endl;
std::cout << finalStr << std::endl;
saveFile();
}
};
int main() {
AffineCipher cipher;
cipher.init();
while (true) {
cipher.showMenu();
cipher.keyDown();
}
return 0;
}摩斯电码
编码表
结构组成
dot 和 dash
。点 / dot / · / 滴 / di
划 / dash / - / 嗒 / dah· 和 - 来编码 hello world
会怎样:
这好像和我们经常见到的不一样,难道不应该是这样的么?
另一个角度,我们知道 - 实际上是三倍长度的 ·。二进制
点 (1)
划 (111)
分隔符
点划分隔 (0)
字符分隔 (000)
单词分隔 (0000000)
可以看到,其实 OR 的摩尔斯编码,其实也可以用二进制 1 和 0 来进行表示:
OR => 010111010001011101010传输效率
信息压缩
摩斯电码对应表参考:https://morsecode.bmcx.com/
常用摩斯电码参考:https://baike.baidu.com/item/%E6%91%A9%E5%B0%94%E6%96%AF%E7%94%B5%E7%A0%81/1527853
斯巴达密码棒
加密
范例文字:"Help me I am under attack".
欲加密:
_________
H E L P M
E I A M U
N D E R A
T T A C K
__________
==>"HENTEIDTLAEAPMRCMUAK"解密
假设编码文为"HENTEIDTLAEAPMRCMUAK"
__________________________
H E N T---------
E I D T---------
L A E A---------
P M R C---------
M U A K---------
_______________
==>"HELPMEIAMUNDERATTACK"波利比奥斯密码
A B C D E F
1 A B C D E F
2 G H I K L M N
3 O P Q R S T U
4 V W X Y Z A 对应 11
B 对应 12
C 对应 13加密
解密
Playfair 密码
加密
解密—解密就是加密的逆过程
c++源码加解密实现
#include "Playfair.h"
string finalStr;
string fileStr;
typedef unordered_map<char, pair<int, int>> myMap;//密码表
myMap m;//5*5密码表 大写
int vis[26];//生成密码表标记 去掉j
void init() {
finalStr = "";
fileStr = "";
m = myMap();
memset(vis, 0, sizeof(vis));
}
void show()
{
cout << "****************Playfair密码****************" << endl;
cout << "\t\t1.加密文件" << endl;
cout << "\t\t2.解密文件" << endl;
cout << "\t\t3.退出" << endl;
cout << "******************************************" << endl;
}
void keyDown()//按键处理
{
int userkey = 0;
cin >> userkey;
switch (userkey) {
case 1:
cout << "-----------------加密文件-----------------" << endl;
readFile();
initkey();
encrypt();
saveFile();
init();
break;
case 2:
cout << "-----------------解密文件-----------------" << endl;
readFile();
initkey();
decrypt();
saveFile();
init();
break;
case 3:
exit(0);
break;
}
}
void readFile()//读取文件
{
cout << "请输入文件名:" << endl;
string fileName;
cin >> fileName;
FILE* fp = fopen(fileName.c_str(), "r+");
if (fp == nullptr) {
cout << "未找到相关文件" << endl;
return;
}
else {
cout << "成功打开文件" << endl;
}
char ch;
int pos = 0;
while ((ch = fgetc(fp)) != EOF) {
fileStr += ch;
}
cout << endl << "明文为:" << endl;
cout << fileStr << endl;
fclose(fp);
}
void saveFile()//保存文件
{
string fileName;
cout << endl << "请输入要保存信息的文件名:" << endl;
cin >> fileName;
FILE* fp = fopen(fileName.c_str(), "w+");
if (fp == nullptr) {
cout << endl << "保存文件失败" << endl;
return;
}
else {
cout << endl << "保存成功" << endl;
}
fprintf(fp, "%s", finalStr.c_str());
fclose(fp);
}
void initkey() {
cout << endl << "请输入密钥以生成密码表:" << endl;
string key = "";//输入密钥
cin >> key;
int i = 0;
int j = 0;
for (char ch : key) {
if (ch == ' ' || vis[ch - 'a'])continue;
if (ch == 'j')ch = 'i';
m.insert(myMap::value_type(ch - 32, { i,j }));
vis[ch - 'a'] = 1;
i += j == 4;
j = (j + 1) % 5;
}
for (int k = 0; k < 26; k++) {
if (k == 9)continue;
if (!vis[k]) {
m.insert(myMap::value_type(k + 'A', { i,j }));
vis[k] = 1;
i += j == 4;
j = (j + 1) % 5;
}
}
//for (auto &temp : m) {
// cout << temp.first << " " << temp.second.first << " " << temp.second.second << endl;
//}
cout << "成功生成密码表" << endl;
}
char researchkey(int x, int y) {//在密码表里面查找明文字母对应的密文字母
for (myMap::iterator it = m.begin(); it != m.end(); it++) {
if (it->second.first == x && it->second.second == y) {
return it->first;
}
}
}
void encrypt() {//整理明文并加密
string temp;//存放大写明文
for (int i = 0; i < fileStr.size(); i++) {// 通过这个for循环去除非法字符以及将明文字母变为大写,将'J'变为'I'
if ((fileStr[i] >= 'a' && fileStr[i] <= 'z') || (fileStr[i] >= 'A' && fileStr[i] <= 'Z')) {
if (fileStr[i] == 'j' || fileStr[i] == 'J') {
temp += 'I';
}
else {
temp += fileStr[i] >= 'a' ? fileStr[i] - 'a' + 'A' : fileStr[i];
}
}
}
if (temp.size() % 2 == 1) {
temp += 'X';
}
//cout << temp << endl;
int j1 = 0, j2 = 0, l1 = 0, l2 = 0; //两个字符的坐标
char p1, p2; //两个字符
for (int i = 0; i < temp.size(); i += 2) {
p1 = temp[i];
p2 = temp[i + 1];
j1 = m[p1].first;
l1 = m[p1].second;
j2 = m[p2].first;
l2 = m[p2].second;
if (p1 != p2) {
//printf("%c %c %d %d %d %d\n",p1,p2,j1,l1,j2,l2);
if (j1 == j2) {
finalStr += researchkey(j1, (l1 + 1) % 5);
finalStr += researchkey(j2, (l2 + 1) % 5);
}
else if (l1 == l2) {
finalStr += researchkey((j1 + 1) % 5, l1);
finalStr += researchkey((j2 + 1) % 5, l2);
}
else if (j1 != j2 && l1 != l2) {
finalStr += researchkey(j1, l2);
finalStr += researchkey(j2, l1);
}
}
else {
//p1p3 p2p3
char p3 = 'X';
int j3, l3;
j3 = m[p3].first;
l3 = m[p3].second;
if (j1 == j3) {
finalStr += researchkey(j1, (l1 + 1) % 5);
finalStr += researchkey(j3, (l3 + 1) % 5);
}
else if (l1 == l3) {
finalStr += researchkey((j1 + 1) % 5, l1);
finalStr += researchkey((j3 + 1) % 5, l3);
}
else if (j1 != j3 && l1 != l3) {
finalStr += researchkey(j1, l3);
finalStr += researchkey(j3, l1);
}
if (j2 == j3) {
finalStr += researchkey(j2, (l2 + 1) % 5);
finalStr += researchkey(j3, (l3 + 1) % 5);
}
else if (l2 == l3) {
finalStr += researchkey((j2 + 1) % 5, l2);
finalStr += researchkey((j3 + 1) % 5, l3);
}
else if (j3 != j2 && l3 != l2) {
finalStr += researchkey(j2, l3);
finalStr += researchkey(j3, l2);
}
}
}
cout << endl << "生成的密文为:" << endl;
cout << finalStr << endl;
}
void decrypt() {//解密密文
string temp = fileStr;
int j1 = 0, j2 = 0, l1 = 0, l2 = 0; //两个字符的坐标
char p1, p2; //两个字符
for (int i = 0; i < temp.size(); i += 2) {
p1 = temp[i];
p2 = temp[i + 1];
j1 = m[p1].first;
l1 = m[p1].second;
j2 = m[p2].first;
l2 = m[p2].second;
if (j1 == j2) {
finalStr += researchkey(j1, (l1 + 4) % 5) + 32;
finalStr += researchkey(j2, (l2 + 4) % 5) + 32;
}
else if (l1 == l2) {
finalStr += researchkey((j1 + 4) % 5, l1) + 32;
finalStr += researchkey((j2 + 4) % 5, l2) + 32;
}
else if (j1 != j2 && l1 != l2) {
finalStr += researchkey(j1, l2) + 32;
finalStr += researchkey(j2, l1) + 32;
}
}
cout << endl << "明文为:" << endl;
cout << finalStr << endl;
}培根密码
培根密码的加密原理
两种加密方式
A aaaaa B aaaab C aaaba D aaabb E aabaa F aabab G aabba H aabbb I abaaa J abaab
K ababa L ababb M abbaa N abbab O abbba P abbbb Q baaaa R baaab S baaba T baabb
U babaa V babab W babba X babbb Y bbaaa Z bbaab a AAAAA g AABBA n ABBAA t BAABA
b AAAAB h AABBB o ABBAB u-v BAABB
c AAABA i-j ABAAA p ABBBA w BABAA
d AAABB k ABAAB q ABBBB x BABAB
e AABAA l ABABA r BAAAA y BABBA
f AABAB m ABABB s BAAAB z BABBB培根密码的解密原理
基于python加解密源码
#!/usr/bin/python
# -*- coding: utf-8 -*-
import re
alphabet = ['a','b','c','d','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','t','u','v','w','x','y','z']
first_cipher = ["aaaaa","aaaab","aaaba","aaabb","aabaa","aabab","aabba","aabbb","abaaa","abaab","ababa","ababb","abbaa","abbab","abbba","abbbb","baaaa","baaab","baaba","baabb","babaa","babab","babba","babbb","bbaaa","bbaab"]
second_cipher = ["aaaaa","aaaab","aaaba","aaabb","aabaa","aabab","aabba","aabbb","abaaa","abaaa","abaab","ababa","ababb","abbaa","abbab","abbba","abbbb","baaaa","baaab","baaba","baabb","baabb","babaa","babab","babba","babbb"]
def encode():
string = raw_input("please input string to encode:\n")
e_string1 = ""
e_string2 = ""
for index in string:
for i in range(0,26):
if index == alphabet[i]:
e_string1 += first_cipher[i]
e_string2 += second_cipher[i]
break
print "first encode method result is:\n"+e_string1
print "second encode method result is:\n"+e_string2
return
def decode():
e_string = raw_input("please input string to decode:\n")
e_array = re.findall(".{5}",e_string)
d_string1 = ""
d_string2 = ""
for index in e_array:
for i in range(0,26):
if index == first_cipher[i]:
d_string1 += alphabet[i]
if index == second_cipher[i]:
d_string2 += alphabet[i]
print "first decode method result is:\n"+d_string1
print "second decode method result is:\n"+d_string2
return
if __name__ == '__main__':
while True:
print "\t*******Bacon Encode_Decode System*******"
print "input should be lowercase,cipher just include a b"
print "1.encode\n2.decode\n3.exit"
s_number = raw_input("please input number to choose\n")
if s_number == "1":
encode()
raw_input()
elif s_number == "2":
decode()
raw_input()
elif s_number == "3":
break
else:
continue猪圈密码
一、密码符号与映射
二、加密过程
三、解密过程
基于python加解密源码
a='abcdefghistuv'
s='jklmnopqrwxyz'
t='Congratulations:ocjp{zkirjwmo-ollj-nmlw-joxi-tmolnrnotvms}'
f=[]
len=len(t)
for i in range(0,len):
n=a.find(t[i])
m=s.find(t[i])
if n==-1 and m==-1:
f.append(t[i])
if n==-1 and m!=-1:
f.append(a[m])
elif m==-1 and n!=-1:
f.append(s[n])
for i in range(0,len):
print(f[i],end="")古典密码的应用场景
古典密码的局限性