java token加密解密算法_java基于Des对称加密算法实现的加密与解密功能详解

本文详细介绍了如何在Java中使用DES对称加密算法进行加密和解密。通过示例代码展示了如何利用`sun.misc.BASE64Decoder`和`sun.misc.BASE64Encoder`以及`org.apache.commons.codec.binary.Base64`进行编码解码。文章还探讨了在编码解码过程中是否需要UTF-8或GBK编码的问题。
摘要由CSDN通过智能技术生成

本文实例讲述了java基于Des对称加密算法实现的加密与解密功能。分享给大家供大家参考,具体如下:

Des 加密相关类介绍:

SecureRandom  这个类是继承自java.util.Random 这个类

SecureRandom 这个类的构造器有三种,下面例举两种:

SecureRandom()构造一个实现默认随机数算法的安全随机数生成器 (RNG)。

SecureRandom(byte[] seed)构造一个实现默认随机数算法的安全随机数生成器 (RNG)。

DESKeySpec 这个类是用来使用原始秘钥来生成秘钥的秘钥内容

DESKeySpec 有两个构造函数:

DESKeySpec(byte[] key) 创建一个 DESKeySpec 对象,使用 key 中的前 8 个字节作为 DES 密钥的密钥内容。

DESKeySpec(byte[] key, int offset) 创建一个 DESKeySpec 对象,使用 key 中始于且包含 offset 的前 8 个字节作为 DES-EDE 密钥的密钥内容。

SecretKeyFactory , 密钥工厂用来将密钥(类型 Key 的不透明加密密钥)转换为密钥规范(底层密钥材料的透明表示形式),反之亦然。秘密密钥工厂只对秘密(对称)密钥进行操作。

SecretKey对象,秘钥对象,通过调用秘钥工厂的generateSecret(DESKeySpec deskeyspace) 方法来生成秘钥

Cipher 类为加密和解密提供密码功能,通过调用Cipher的getInstance("des") 来获取实例

Cipher 对象调用init() 方法进行对象的初始化,init() 方法的具体参数按照具体情况而定,有加密的也有解密的常量

最后调用Cipher的doFinal() 方法进行加密解密。

在这里请教大家一个问题,不管是第一种使用BASE64Encoder编码还是第二种org.apache.commons.codec.binary.Base64编码,在将String 转化为byte以及将byte转化为String 时需要 UTF-8/GBK 等编码来编码,解码吗?

一、使用了 sun.misc.BASE64Decoder 和BASE64Encoder 进行解码,编码

package com.soufun.com;

import java.io.IOException;

import java.security.NoSuchAlgorithmException;

import java.security.SecureRandom;

import java.util.Date;

import javax.crypto.Cipher;

import javax.crypto.SecretKey;

import javax.crypto.SecretKeyFactory;

import javax.crypto.spec.DESKeySpec;

// 导入sun的64位编码

import sun.misc.BASE64Decoder;

import sun.misc.BASE64Encoder;

/**

*@author WHD

*

*即使导入sun.misc这个架包也会报错,这时首先把你的JRE架包移除再导入一次就可以了

*/

public class DesUtil {

// 定义加密方式

private final static String DES = "DES";

private final static String UTF8="GBK";

static SecretKeyFactory keyFactory = null;

static {

try {

keyFactory=SecretKeyFactory.getInstance("DES");

} catch (NoSuchAlgorithmException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

public static void main(String[] args) throws Exception {

long begin=new Date().getTime();

String data = "aaades加密测试";

// 注意:DES加密和解密过程中,密钥长度都必须是8的倍数

String key = "qazwsxed";

System.err.println(encrypt(data, key));

System.err.println(decrypt(encrypt(data, key), key));

long end =new Date().getTime();

System.out.println(end-begin);

}

/**

* Description 根据键值进行加密

* @param data

* @param key 加密键byte数组

* @return

* @throws Exception

*/

public static String encrypt(String data, String key) throws Exception {

// 使用指定的编码获取要加密的内容,一般秘钥都是字母或数字不用指定编码,但指定也可以

byte[] bt = encrypt(data.getBytes(UTF8), key.getBytes(UTF8));

//注意:在加密和解密的时候使用sun的BASE64Encoder()进行编码和解码不然会有乱码

//网上查看了很多实例,都没有编码和解码,也说没有乱码问题,而我这里出现了乱码,所以使用BASE64Encoder()进行了编码解码

String strs = new BASE64Encoder().encode(bt);

return strs;

}

/**

* Description 根据键值进行解密

* @param data

* @param key 加密键byte数组

* @return

* @throws IOException

* @throws Exception

*/

public static String decrypt(String data, String key) throws IOException,

Exception {

if (data == null)

return null;

//注意:在加密和解密的时候使用sun的BASE64Encoder()进行编码和解码不然会有乱码

BASE64Decoder decoder = new BASE64Decoder();

byte[] buf = decoder.decodeBuffer(data);

byte[] bt = decrypt(buf,key.getBytes());

return new String(bt,UTF8);

}

/**

* Description 根据键值进行加密

* @param data

* @param key 加密键byte数组

* @return

* @throws Exception

*/

private static byte[] encrypt(byte[] data, byte[] key) throws Exception {

// 生成一个可信任的随机数源

SecureRandom sr = new SecureRandom();

// 从原始密钥数据创建DESKeySpec对象,也就是创建秘钥的秘钥内容

DESKeySpec dks = new DESKeySpec(key);

// 密钥工厂用来将密钥(类型 Key 的不透明加密密钥)转换为密钥规范(底层密钥材料的透明表示形式),反之亦然。秘密密钥工厂只对秘密(对称)密钥进行操作。

// 这里改为使用单例模式

//SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(DES);

//根据提供的密钥规范(密钥材料)生成 SecretKey(秘钥) 对象。

SecretKey securekey = keyFactory.generateSecret(dks);

// Cipher对象实际完成加密操作,此类为加密和解密提供密码功能

Cipher cipher = Cipher.getInstance(DES);

// 用密钥和随机源初始化此 Cipher。ENCRYPT_MODE用于将 Cipher 初始化为加密模式的常量。

cipher.init(Cipher.ENCRYPT_MODE, securekey, sr);

//正式执行加密操作

return cipher.doFinal(data);

}

/**

* Description 根据键值进行解密

* @param data

* @param key 加密键byte数组

* @return

* @throws Exception

*/

private static byte[] decrypt(byte[] data, byte[] key) throws Exception {

// 生成一个可信任的随机数源

SecureRandom sr = new SecureRandom();

// 从原始密钥数据创建DESKeySpec对象,也就是创建秘钥的秘钥内容

DESKeySpec dks = new DESKeySpec(key);

// 密钥工厂用来将密钥(类型 Key 的不透明加密密钥)转换为密钥规范(底层密钥材料的透明表示形式),反之亦然。秘密密钥工厂只对秘密(对称)密钥进行操作。

// 这里改为使用单例模式

//SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(DES);

//根据提供的密钥规范(密钥材料)生成 SecretKey(秘钥)对象。

SecretKey securekey = keyFactory.generateSecret(dks);

// Cipher类为加密和解密提供密码功能

Cipher cipher = Cipher.getInstance(DES);

// DECRYPT_MODE用于将 Cipher 初始化为解密模式的常量。

cipher.init(Cipher.DECRYPT_MODE, securekey, sr);

// 正式进行解密操作

return cipher.doFinal(data);

}

}

二、使用org.apache.commons.codec.binary.Base64 进行解码,编码

package com.soufun.com;

import java.io.IOException;

import java.security.NoSuchAlgorithmException;

import java.security.SecureRandom;

import java.util.Date;

import javax.crypto.Cipher;

import javax.crypto.SecretKey;

import javax.crypto.SecretKeyFactory;

import javax.crypto.spec.DESKeySpec;

import org.apache.commons.codec.binary.Base64;

/**

*@author WHD

*

*/

public class DesUtil {

// 定义加密方式

private final static String DES = "DES";

private final static String UTF8="GBK";

static SecretKeyFactory keyFactory = null;

static {

try {

keyFactory=SecretKeyFactory.getInstance("DES");

} catch (NoSuchAlgorithmException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

public static void main(String[] args) throws Exception {

long begin=new Date().getTime();

String data = "aaades加密测试";

// 注意:DES加密和解密过程中,密钥长度都必须是8的倍数

String key = "qazwsxed";

System.err.println(encrypt(data, key));

System.err.println(decrypt(encrypt(data, key), key));

long end =new Date().getTime();

System.out.println(end-begin);

}

/**

* Description 根据键值进行加密

* @param data

* @param key 加密键byte数组

* @return

* @throws Exception

*/

public static String encrypt(String data, String key) throws Exception {

// 使用指定的编码获取要加密的内容,一般秘钥都是字母或数字不用指定编码,但指定也可以

byte[] bt = encrypt(data.getBytes(UTF8), key.getBytes());

// 第一个使用了sun.misc.BASE64Encoder;进行了编码,但网上说使用org.apache.commons.codec.binary.Base64比较好所以拿来试试

String strs = Base64.encodeBase64String(bt);

return strs;

}

/**

* Description 根据键值进行解密

* @param data

* @param key 加密键byte数组

* @return

* @throws IOException

* @throws Exception

*/

public static String decrypt(String data, String key) throws IOException,

Exception {

if (data == null)

return null;

// 使用org.apache.commons.codec.binary.Base64解码

byte [] buf=Base64.decodeBase64(data);

byte[] bt = decrypt(buf,key.getBytes());

return new String(bt,UTF8);

}

/**

* Description 根据键值进行加密

* @param data

* @param key 加密键byte数组

* @return

* @throws Exception

*/

private static byte[] encrypt(byte[] data, byte[] key) throws Exception {

// 生成一个可信任的随机数源

SecureRandom sr = new SecureRandom();

// 从原始密钥数据创建DESKeySpec对象,也就是创建秘钥的秘钥内容

DESKeySpec dks = new DESKeySpec(key);

// 密钥工厂用来将密钥(类型 Key 的不透明加密密钥)转换为密钥规范(底层密钥材料的透明表示形式),反之亦然。秘密密钥工厂只对秘密(对称)密钥进行操作。

// 这里改为使用单例模式

//SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(DES);

//根据提供的密钥规范(密钥材料)生成 SecretKey(秘钥) 对象。

SecretKey securekey = keyFactory.generateSecret(dks);

// Cipher对象实际完成加密操作,此类为加密和解密提供密码功能

Cipher cipher = Cipher.getInstance(DES);

// 用密钥和随机源初始化此 Cipher。ENCRYPT_MODE用于将 Cipher 初始化为加密模式的常量。

cipher.init(Cipher.ENCRYPT_MODE, securekey, sr);

//正式执行加密操作

return cipher.doFinal(data);

}

/**

* Description 根据键值进行解密

* @param data

* @param key 加密键byte数组

* @return

* @throws Exception

*/

private static byte[] decrypt(byte[] data, byte[] key) throws Exception {

// 生成一个可信任的随机数源

SecureRandom sr = new SecureRandom();

// 从原始密钥数据创建DESKeySpec对象,也就是创建秘钥的秘钥内容

DESKeySpec dks = new DESKeySpec(key);

// 密钥工厂用来将密钥(类型 Key 的不透明加密密钥)转换为密钥规范(底层密钥材料的透明表示形式),反之亦然。秘密密钥工厂只对秘密(对称)密钥进行操作。

// 这里改为使用单例模式

//SecretKeyFactory keyFactory = SecretKeyFactory.getInstance(DES);

//根据提供的密钥规范(密钥材料)生成 SecretKey(秘钥)对象。

SecretKey securekey = keyFactory.generateSecret(dks);

// Cipher类为加密和解密提供密码功能

Cipher cipher = Cipher.getInstance(DES);

// DECRYPT_MODE用于将 Cipher 初始化为解密模式的常量。

cipher.init(Cipher.DECRYPT_MODE, securekey, sr);

// 正式进行解密操作

return cipher.doFinal(data);

}

}

一、二中使用到的架包下载地址:

三、未使用任何编码,解码架包

package com.soufun.com;

import java.io.IOException;

import java.security.NoSuchAlgorithmException;

import java.util.Date;

import java.util.HashMap;

import java.util.Map;

import javax.crypto.Cipher;

import javax.crypto.SecretKey;

import javax.crypto.SecretKeyFactory;

import javax.crypto.spec.DESKeySpec;

import javax.crypto.spec.IvParameterSpec;

/**

*@author WHD

*

*/

public class DESCrypt {

static SecretKeyFactory secretKeyFactory = null;

//Cipher 的“算法/模式/填充”

static final String CIPHER = "DES/CBC/PKCS5Padding";

static {

try {

// 在静态代码块中获取秘钥工程

secretKeyFactory = SecretKeyFactory.getInstance("DES");

} catch (NoSuchAlgorithmException e) {

e.printStackTrace();

}

}

// 定义常量 ,编码格式

private static final String UTF8 = "GBK";

/*

* 对象缓存的容器

*/

static abstract class Cache {

private final Map innerCache = new HashMap();

protected abstract Object createValue(Object key) throws Exception;

public Object get(Object key) throws Exception {

Object value;

synchronized (innerCache) {

value = innerCache.get(key);

if (value == null) {

value = new CreationPlaceholder();

innerCache.put(key, value);

}

}

if (value instanceof CreationPlaceholder) {

synchronized (value) {

CreationPlaceholder progress = (CreationPlaceholder) value;

if (progress.value == null) {

progress.value = createValue(key);

synchronized (innerCache) {

innerCache.put(key, progress.value);

}

}

return progress.value;

}

}

return value;

}

static final class CreationPlaceholder {

Object value;

}

}

/*

* hex->str & str->hex

*/

public static byte[] stringToHex(String ss) {

// 字符串转化we

byte digest[] = new byte[ss.length() / 2];

for (int i = 0; i < digest.length; i++) {

String byteString = ss.substring(2 * i, 2 * i + 2);

int byteValue = Integer.parseInt(byteString, 16);

digest[i] = (byte) byteValue;

}

return digest;

}

public static String hexToString(byte b[]) {

StringBuffer hexString = new StringBuffer();

for (int i = 0; i < b.length; i++) {

String plainText = Integer.toHexString(0xff & b[i]);

if (plainText.length() < 2) {

hexString.append("0");

}

hexString.append(plainText);

}

return hexString.toString();

}

private static byte[] _convertKeyIv(String text) throws IOException {

if (text.length() == 8) {

return text.getBytes(UTF8);

}

if (text.startsWith("0x") && text.length() == 32) {

byte[] result = new byte[8];

for (int i = 0; i < text.length(); i += 2) {

if (text.charAt(i++) == '0' && text.charAt(i++) == 'x') {

try {

result[i / 4] = (byte) Integer.parseInt(

text.substring(i, i + 2), 16);

} catch (Exception e) {

throw new IOException("TXT '" + text + "' is invalid!");

}

}

}

return result;

}

throw new IOException("TXT '" + text + "' is invalid!");

}

/*

* SecretKey & IvParameterSpec的缓存

*/

private static Cache SecretKeySpecs = new Cache() {

protected Object createValue(Object key) throws Exception {

SecretKey secretKeyObj = null;

try {

secretKeyObj = secretKeyFactory.generateSecret(new DESKeySpec(

_convertKeyIv((String) key)));

} catch (Exception e) {

e.printStackTrace();

}

return secretKeyObj;

}

};

private static Cache IvParamSpecs = new Cache() {

protected Object createValue(Object key) throws Exception {

IvParameterSpec ivObj = null;

ivObj = new IvParameterSpec(_convertKeyIv((String) key));

return ivObj;

}

};

/*

* 加密&解密

*/

public static String encrypt(String text, String authKey, String authIv) {

SecretKey secretKeyObj = null;

IvParameterSpec ivObj = null;

try {

secretKeyObj = (SecretKey) SecretKeySpecs.get(authKey);

ivObj = (IvParameterSpec) IvParamSpecs.get(authIv);

} catch (Exception e) {

e.printStackTrace();

}

byte[] data = null;

try {

data = text.getBytes(UTF8);

} catch (Exception e) {

e.printStackTrace();

}

byte[] authToken = null;

try {

authToken = encrypt(data, secretKeyObj, ivObj);

} catch (Exception e) {

e.printStackTrace();

}

return hexToString(authToken);

}

public static byte[] encrypt(byte[] data, SecretKey secretKey,

IvParameterSpec iv) throws Exception {

Cipher cipher = Cipher.getInstance(CIPHER);

cipher.init(Cipher.ENCRYPT_MODE, secretKey, iv);

return cipher.doFinal(data);

}

public static String decrypt(String hexString, String authKey, String authIv)

throws Exception {

SecretKey secretKeyObj = null;

IvParameterSpec ivObj = null;

try {

secretKeyObj = (SecretKey) SecretKeySpecs.get(authKey);

ivObj = (IvParameterSpec) IvParamSpecs.get(authIv);

} catch (Exception e) {

e.printStackTrace();

}

String text = decrypt(hexString, secretKeyObj, ivObj);

return text;

}

public static String decrypt(String message, SecretKey secretKey,

IvParameterSpec iv) throws Exception {

byte[] data = stringToHex(message);

return decrypt(data, secretKey, iv);

}

public static String decrypt(byte[] data, SecretKey secretKey,

IvParameterSpec iv) throws Exception {

Cipher cipher = Cipher.getInstance(CIPHER);

cipher.init(Cipher.DECRYPT_MODE, secretKey, iv);

byte[] retByte = cipher.doFinal(data);

return new String(retByte);

}

public static void main(String[] args) throws Exception {

long begin= new Date().getTime();

String authKey = "w8f3k9c2";

String authIv = "w8f3k9c2";

String text = "aaades加密测试";

// 140CB412BA03869F

// 140cb412ba03869f

// 对原文进行加密

String encryptedText = encrypt(text, authKey, authIv);

System.out.println("encryptedText:" + encryptedText);

// 对密文进行还原

String plainText = decrypt(encryptedText, authKey, authIv);

System.out.println("plainText:" + plainText);

//2a329740ce15f549be64190b183a5be2

long end =new Date().getTime();

System.out.println(end-begin);

}

}

PS:关于加密解密感兴趣的朋友还可以参考本站在线工具:

在线MD5/hash/SHA-1/SHA-2/SHA-256/SHA-512/SHA-3/RIPEMD-160加密工具:http://tools.jb51.net/password/hash_md5_sha

希望本文所述对大家java程序设计有所帮助。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值
>