JavaScript中的Caesar Cipher:密码已破解
Caesar Cipher in javascript: cipher broken
我正在用javascript构建一个Caesar Cipher。它使用一个名为currentkey
的随机设置变量作为密码的密钥。它可以是 -25 到 +25 之间的数字,跳过 0。
我无法弄清楚为什么该函数在字符串的开头返回 undefined,或者为什么只要字符串存在,它就会翻译相同的字母,甚至为什么该字母根本没有被翻译。
var currentkey = 5 //for example
function convertKey(str) {
var o_text = str;
var o_letters = o_text.split("");
var alphabet = ['a','b','c','d','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','t','u','v','x','y','w','z']
var c_text, _x = "";
for (x in o_letters) {
if (currentkey > 0) {
_x = alphabet[alphabet.indexOf(x) + currentkey]
} else {
_x = alphabet[alphabet.indexOf(x) + (26 - currentkey)]
}
c_text = c_text + _x;
}
return c_text;
}
例如,运行convertKey("example")
返回undefinedeeeeeee
(未定义 + "example"中第一个字母的 7 倍)。
实现 JavaScript 中的 Caesar Cipher 算法非常有趣,因为 JavaScripts 缺乏真正的模运算符。%只是对除法的提醒。阅读本文以获取更多解释。
但是,您可以轻松地将模定义为自定义函数,然后继续实施凯撒密码 - 这是一种非常简单的加密形式,其中原始消息中的每个字母都向左或向右移动一定数量的位置。
要解密消息,我们只需将字母移回相同数量的位置。
例:
- JAVASCRIPT 变成 MDYDVFULSW 如果我们将所有字母移动 3 个位置
- MDYDVFULSW 如果我们将所有字母后移 3 个位置,则会返回 JAVASCRIPT。
如果移动后一个字母超出了字母的范围,那么这个字母就会被字母包裹起来。示例:如果字母 Z 移动 3 个位置,则变为 C。
这种"环绕"效果意味着使用模数。用数学术语来说,以上可以表示为:
En(x) = (x + n) mod 26
Dn(x)= (x – n) mod 26
尝试在不使用适当的模运算符的情况下在 JavaScript 中实现此算法将产生不正确的结果或非常神秘且难以理解的代码。
通过使用自定义模函数,代码以相同的方式表示数学方程:
// Function will implement Caesar Cipher to
// encrypt / decrypt the msg by shifting the letters
// of the message acording to the key
function encrypt(msg, key)
{
var encMsg = "";
for(var i = 0; i < msg.length; i++)
{
var code = msg.charCodeAt(i);
// Encrypt only letters in 'A' ... 'Z' interval
if (code >= 65 && code <= 65 + 26 - 1)
{
code -= 65;
code = mod(code + key, 26);
code += 65;
}
encMsg += String.fromCharCode(code);
}
return encMsg;
}
// Implement modulo by replacing the negative operand
// with an equivalent positive operand that has the same wrap-around effect
function mod(n, p)
{
if ( n < 0 )
n = p - Math.abs(n) % p;
return n % p;
}
玩得愉快!
加密一些消息以试用代码。请记住:如果使用正密钥进行加密,请使用互补的负密钥对其进行解密。您还可以使用此代码解码出现在 Web 和新闻组上随处可见的 ROT13 消息。
如果您想找出在 JavaScript 中实现模的其他方法,请参阅本文开头提到的文章。
undefined
是在尝试将其与 _x
连接之前未初始化c_text
的结果。
该代码仅适用于一个字母,因为alphabet.indexOf(x)
返回-1
(未找到)。使用 o_text = "abc"
时,x
等于 0、1 和 2。因此,alphabet
中的索引 0、1 或 2 不存在(由-1
结果指示)。您需要将这些数字与o_text
相关联,方法是将indexOf(x)
更改为indexOf(o_text[x])
。此外,为了防止超出数组的边界,您需要一个模数运算来将大于 26 的值包装回有效区域(我使用了 alphabet.length
来允许此代码与其他字母一起使用)。因此,正确的代码如下(注意:我已经将数组中的"w"按字母顺序移动到其正确的位置,因为我假设它在您的示例中的位置是一个错误而不是故意的):
var currentkey = 5 //for example
function convertKey(str) {
var o_text = str;
var o_letters = o_text.split("");
var 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']
var c_text="", _x = "";
for (x in o_letters) {
if (currentkey > 0) {
_x = alphabet[(alphabet.indexOf(o_letters[x]) + currentkey) % alphabet.length]
} else {
_x = alphabet[(alphabet.indexOf(o_letters[x]) + (26 - currentkey)) % alphabet.length]
}
c_text = c_text + _x;
}
return c_text;
}
alert(convertKey('abcdefghijklmnopqrstuvwxyz'));
此警报的fghijklmnopqrstuvwxyzabcde
(1) 您没有正确遍历数组o_letters
。
(2)你的阵列越界了。
for (var i = 0; i < o_letters.length; i++) {
_x = alphabet[(alphabet.indexOf(o_letters[i]) + currentkey + 26) % 26]
c_text = c_text + _x;
}
此外,在您的代码中,您根本不需要.split("")
。
我只会这样做:
var currentkey = 5 //for example
function convertKey(str) {
var ret = '';
for (var i = 0; i < str.length; i++) {
ret += String.fromCharCode((str.charCodeAt(i) + currentKey + 26) % 26);
}
return ret;
}
或者更简洁(但效率较低):
function convertKey(str) {
return str.split('').map(function(c) {
return String.fromCharCode((c.charCodeAt(0) + currentKey + 26) % 26);
}).join('');
}
//在 https://www.freecodecamp.org/challenges/caesars-cipher 工作
function rot13(str){
var res = [];
var currentPosition;
var shiftedPosition;
for (var i = 0; i<str.length; i++){
currentPosition = str.charCodeAt(i);
if (currentPosition<65 || currentPosition>90){
res.push(String.fromCharCode(currentPosition));
}
shiftedPosition = str.charCodeAt(i) - 13;
if (currentPosition>=65 && shiftedPosition <65){
res.push(String.fromCharCode(91-(13-(currentPosition-65))));
}
if (currentPosition>=78 && currentPosition<=90){
res.push(String.fromCharCode(shiftedPosition));
}
}
return res.join('');`enter code here`
}
// Change the inputs below to test
rot13("GUR DHVPX OEBJA QBT WHZCRQ BIRE GUR YNML SBK.");
这是一个更简单的答案:
var rot = {
'A': 'N',
'B': 'O',
'C': 'P',
'D': 'Q',
'E': 'R',
'F': 'S',
'G': 'T',
'H': 'U',
'I': 'V',
'J': 'W',
'K': 'X',
"L": 'Y',
'M': 'Z',
'N': 'A',
'O': 'B',
'P': 'C',
'Q': 'D',
'R': 'E',
'S': 'F',
'T': 'G',
'U': 'H',
'V': 'I',
'W': 'J',
'X': 'K',
'Y': 'L',
'Z': 'M',
' ': ' ',
'.': '.',
'!': '!',
'?': '?'
};
// Change the inputs below to test
rot13("SERR CVMMN!");
// FREE CODE CAMP
function rot13(str) {
var index = [];
for (var key in str) {
for (var property in rot) {
if (str[key] === property) {
index.push(rot[property]);
}
}
}
return index.join('');
}
function rot13(str) {
const alphabet1 = 'ABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXY';
const alphabet = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
return str
.split('')
.map(function(char) {
const pos = alphabet.indexOf(char);
return pos >=0? alphabet1[pos+13] : char;
})
.join('');
}
rot13("SERR PBQR PNZC");
- 如何使用密码检测网络中的状态连接
- Meteor忘记了密码的实现
- 使用密码对话框Javascript请求帮助
- Internet Explorer缺少占位符支持,特别是密码字段
- angular js密码强度显示问题
- 一个密码测试程序,如果存在空格,它会提醒用户
- 密码输入键脚本在首次使用后无法工作
- JavaScript生成html格式的密码
- 在表单完成并确认密码之前,请禁用提交按钮
- pushState()正在驳回'记住密码'对话
- 如何对单个页面进行密码保护(这不是身份验证系统的一部分)
- AngularJs Cordova安卓应用程序中输入的4位密码PIN验证
- 在节点 js 中使用护照本地模块验证(登录)密码时出错
- 难以置信的简单密码系统不工作
- 密码字段的问题
- 在每个字符输入后,使密码文本框值可见
- PassportJS/忘记密码:req.user只定义一次
- 如何在ajax中检查密码是否小于8个字符
- 访问下一个html页面的静态4位密码
- JavaScript中的Caesar Cipher:密码已破解