JavaScript的加密和解密用的是google的CryptoJS库。本文以AES/ECB/NoPadding为例展示AES加密和解密的方法。

需要下载CryptoJS库,下载地址如下:
https://github.com/sytelus/CryptoJS

需要引入库文件:
<script src="./CryptoJS-master/rollups/aes.js"></script>
<script src="./CryptoJS-master/components/mode-ecb.js"></script>
<script src="./CryptoJS-master/components/pad-nopadding.js"></script>

加密方法:

function encrypt(str) {
var decArray = hexStrToDecArray(str);
var wordArray = int8parse(decArray);
var encrypted = CryptoJS.AES.encrypt(wordArray, aesKeyBytes(), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.NoPadding
});
return wordArrayToHexStr(encrypted.ciphertext.words);
}

这个方法传入的是十六进制字符串,格式为"AA BB CC DD EE FF";
hexStrToDecArray()方法用来将十六进制字符串转换成十进制数字的数组;
int8parse()方法用来将十进制数字转换成CryptoJS所需要的wordArray数组;
最后,encrypt()方法加密得到的是一个wordArray数组,wordArrayToHexStr()再将其解析成十六进制字符串。

解密方法:

function decrypt(str) {
var decArray = hexStrToDecArray(str);
var wordArray = int8parse(decArray);
var base64Str = CryptoJS.enc.Base64.stringify(wordArray);
var decrypted = CryptoJS.AES.decrypt(base64Str, aesKeyBytes(), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.NoPadding
});
return wordArrayToHexStr(decrypted.words);
}

解密方法同加密方法类似,但是CryptoJS.AES.decrypt()方法需要传入Base64格式的加密字符串,因此这里需要先使用CryptoJS.enc.Base64.stringify()方法做一次转换。

以下是完整的Demo代码:

<!DOCTYPE html>
<html>
<head>
<script src="./CryptoJS-master/rollups/aes.js"></script>
<script src="./CryptoJS-master/components/mode-ecb.js"></script>
<script src="./CryptoJS-master/components/pad-nopadding.js"></script>
<script>
function encryptText() {
var plain = document.getElementById("plain").value;
console.log("plain: " + plain);
var encrypted = encrypt(plain);
console.log("encrypted: " + encrypted);
document.getElementById("encrypted").value = encrypted;
} function decryptText() {
var encrypted = document.getElementById("todecrypt").value;
console.log("encrypted: " + encrypted);
var decrypted = decrypt(encrypted);
console.log("decrypted: " + decrypted);
document.getElementById("decrypted").value = decrypted;
} // 加解密用到的密钥
function aesKeyBytes() {
var key_Int = new Int8Array([65, 144, 48, 53, 18, 52, 86, 120, 131, 116, 124, 139, 237, 203, 169, 135]);
var keyBytes = int8parse(key_Int);
return keyBytes;
} // 十六进制字符串数组,个数如果不足16整数倍则补0
function hexTo16Hex(str) {
var diff = 16 - (str.length + 1) / 3 % 16;
for(var i = 0; i < diff; i++) {
str = str + " 00";
}
return str;
} // AES加密
function encrypt(str) {
var decArray = hexStrToDecArray(str);
var wordArray = int8parse(decArray);
var encrypted = CryptoJS.AES.encrypt(wordArray, aesKeyBytes(), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.NoPadding
});
return wordArrayToHexStr(encrypted.ciphertext.words);
} // AES解密
function decrypt(str) {
var decArray = hexStrToDecArray(str);
var wordArray = int8parse(decArray);
var base64Str = CryptoJS.enc.Base64.stringify(wordArray);
var decrypted = CryptoJS.AES.decrypt(base64Str, aesKeyBytes(), {
mode: CryptoJS.mode.ECB,
padding: CryptoJS.pad.NoPadding
});
return wordArrayToHexStr(decrypted.words);
} // 构建WordArray对象
function int8parse(u8arr) {
var len = u8arr.length;
var words = [];
for (var i = 0; i < len; i++) {
words[i >>> 2] |= (u8arr[i] & 0xff) << (24 - (i % 4) * 8);
}
return CryptoJS.lib.WordArray.create(words, len);
} // 十六进制字符串(空格分割)转成十进制数字的数组
function hexStrToDecArray(str) {
var strArray = str.split(" ");
var decArray = [];
for(var i = 0; i < strArray.length; i++) {
decArray.push(parseInt(strArray[i], 16));
}
return arrayTo16Array(decArray);
} // 十进制数组转成十六进制字符串
function decArrayToHexStr(array) {
var hexStr = "";
for(var i = 0; i < array.length; i++) {
var str = array[i].toString(16).toUpperCase();
if (str.length < 2) {
str = "0" + str;
}
hexStr = hexStr + str + " ";
}
return hexStr.substr(0, hexStr.length - 1);
} // word类型的十进制数组转成十六进制字符串
function wordArrayToHexStr(array) {
var hexStr = "";
for(var i = 0; i < array.length; i++) {
var num = array[i];
if (num < 0) {
num = array[i] + 0x100000000;
}
var str = num.toString(16).toUpperCase();
var fullStr = str;
if (str.length < 8) {
for(var j = 0; j < 8 - str.length; j++) {
fullStr = "0" + fullStr;
}
}
hexStr = hexStr + fullStr;
}
var ret = "";
for(var i = 0; i < hexStr.length; i += 2) {
ret = ret + hexStr.substr(i, 2) + " "
}
return ret.substr(0, ret.length - 1);
} // 数组元素个数必须是16的整数倍,不足的在后面补0
function arrayTo16Array(array) {
var len = array.length;
var distLen = parseInt((array.length - 1) / 16) * 16 + 16;
for(var i = array.length; i < distLen; i++) {
array[i] = 0;
}
return array;
}
</script>
</head>
<body>
<h1>加解密测试</h1>
AES128,ECB模式,PaddingModeZeros,不足部分补0
<br>
<input id="plain" type="text" style="width:500px; height:20px;"/>
<button type="button" onclick="encryptText()">加密</button>
<input id="encrypted" type="text" style="width:500px; height:20px;" />
<br>
<br>
<br>
<input id="todecrypt" type="text" style="width:500px; height:20px;"/>
<button type="button" onclick="decryptText()">解密</button>
<input id="decrypted" type="text" style="width:500px; height:20px;" />
</body>
</html>

在浏览器中打开之后效果如下:

输入要加密或解密的字符串,点击加密/解密按钮即可。

JavaScript实现AES算法加密和解密的更多相关文章

  1. ORACLE 字段AES算法加密、解密

    ORACLE 字段AES算法加密.解密(解决中文乱码问题)2014年02月12日 17:13:37 华智互联 阅读数:97971.加解密函数入口 CREATE OR REPLACE FUNCTION ...

  2. JAVA实现AES的加密和解密算法

    原文 JAVA实现AES的加密和解密算法 import javax.crypto.Cipher; import javax.crypto.spec.IvParameterSpec; import ja ...

  3. C#与Java互通AES算法加密解密

    /// <summary>AES加密</summary> /// <param name="text">明文</param> /// ...

  4. .NET与Java互通AES算法加密解密

    /// <summary>AES加密</summary> /// <param name="text">明文</param> /// ...

  5. AES对称加密和解密

    package demo.security; import java.io.IOException; import java.io.UnsupportedEncodingException; impo ...

  6. java的AES对称加密和解密,有偏移量

    import java.math.BigDecimal; import javax.crypto.Cipher; import javax.crypto.spec.IvParameterSpec; i ...

  7. AES对称加密和解密(转)

    AES对称加密和解密 package demo.security; import java.io.IOException; import java.io.UnsupportedEncodingExce ...

  8. AES块加密与解密

    AES块加密与解密 解密目标 在CBC和CTR两种模式下分别给出十篇加密的样例密文,求解密一篇特定的密文 解密前提 全部密文及其加密使用的key都已给出 加密的方法遵循AES的标准 解密过程分析 实验 ...

  9. java sm4国密算法加密、解密

      java sm4国密算法加密.解密 CreationTime--2018年7月5日09点20分 Author:Marydon 1.准备工作 所需jar包: bcprov-jdk15on-1.59. ...

随机推荐

  1. bzoj 1084: [SCOI2005]最大子矩阵【dp】

    分情况讨论,m=1的时候比较简单,设f[i][j]为到i选了j个矩形,前缀和转移一下就行了 m=2,设f[i][j][k]为1行前i个,2行前j个,一共选了k个,i!=j的时候各自转移同m=1,否则转 ...

  2. Spring Boot之配置文件值注入(@ConfigurationProperties)

    前言:Spring Boot配置文件值的注入有两种方式,分别是 @ConfigurationProperties @Value 这里我们使用第一种 首先我们创建一个application.yml文件, ...

  3. Python中处理日期时间库的使用方法(转载)

    <本文来自公众号“大邓带你玩python”,转载> 用百分之20时间,学会解决百分之80的问题. 常用的库有time.datetime.其中datetime库是对time库的封装,所以使用 ...

  4. VS2010创建C++静态链接库创建和使用

    VS2010创建C++静态链接库的方法: 1. 创建一个新项目,在已安装的模板中选择“常规”,在右边的类型下选择“空项目”,在名称和解决方案名称中输入 staLIB.点击确定. 2.在解决方案资源管理 ...

  5. Windows中句柄和ID的区别

    写在前面:这里介绍句柄 对于“句柄”,在下一直停留在一知半解的认识层面,近日在下学习Windows编程,决定趁此机会将句柄彻底搞清楚.查阅了一些网络上的资料,发现网络上的讲解大概可以分为两类:一种是以 ...

  6. UCOSII学习 - 创建任务

    本人刚刚学习UCOSII,平台为正点原子的STM32F103战舰开发板,写这篇博客主要是为了学习UCOSII,也方便自己能够一点一点的进步,话不多说直入正题吧. 第一步:在STM32上移植好UCOSI ...

  7. hdu 4565 So Easy! (共轭构造+矩阵快速幂)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4565 题目大意: 给出a,b,n,m,求出的值, 解题思路: 因为题目中出现了开根号,和向上取整后求 ...

  8. 计算机视觉-SIFT特征匹配进行目标转换

    Lowe将SIFT算法分解为如下四步: 1. 尺度空间极值检测:搜索所有尺度上的图像位置.通过高斯微分函数来识别潜在的对于尺度和旋转不变的兴趣点. 关键点定位:在每个候选的位置上,通过一个拟合精细的模 ...

  9. jdk 1.8下 java ArrayList 添加元素解析

    转载请注明http://www.cnblogs.com/majianming/p/8006452.html 有人问我,java ArrayList底层是怎么实现的?我就回答数组,他再问我,那它是怎么实 ...

  10. 好用的SqlParamterList

    public class SqlParameterList : List<SqlParameter> { #region Properties /// <summary> // ...