我必须在我的服务器上解密一个帧。加密帧来自客户端设备通过GPRS在socket上。加密是通过“TripleDes”完成的,并且使用给定的key.same算法和密钥我使用n服务器端。 frame是Hex和Ascii String的组合。问题是当我解密这个框架时我得到一个错误:
javax.crypto.IllegalBlockSizeException: Input length must be multiple of 8 when decrypting with padded cipher
当我用
之类的零填充我的字节数组时temparray[87] = 00;
我收到错误:
javax.crypto.BadPaddingException: Given final block not properly padded
以下是我的代码:
if ((len = inputStream.read(mainBuffer)) > -1) {
totalLength = len;
}
if (totalLength > 0) {
byteToAscii = function.byteToAscii(mainBuffer, totalLength);
}
if (byteToAscii.length() > 0) {
completeHexString = function.stringToHex(byteToAscii);
debugInfo = "FRAME RECV.=" + completeHexString;
/* FRAME RECV.=41ed34a41a9de6d270aa1e1464527e88c8bee66a00cfb308f60c105de81db0f1ce43d8c0b9bc4e8070b5ab8d4d3650b55d23223fc687bb1485945bc3228e9707a7aecda9f90657e0ac009571c6469c58a2cd9793cc433ccb5993f2*/
}
byte[] key = new byte[]{31, 30, 31, 36, 32, 11, 11, 11, 22, 26, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30};
myKeySpec = new DESedeKeySpec(key);
mySecretKeyFactory = SecretKeyFactory.getInstance("TripleDES");
dekey = mySecretKeyFactory.generateSecret(myKeySpec);
byte[] zeros = {0, 0, 0, 0, 0, 0, 0, 0};
IvParameterSpec iv = new IvParameterSpec(zeros);
Cipher c = Cipher.getInstance("TripleDES/CBC/PKCS5Padding");
c.init(Cipher.DECRYPT_MODE, key, iv);
byte[] decordedValue = new BASE64Decoder().decodeBuffer(completeHexString);
byte[] decValue = c.doFinal(decordedValue);
String decryptedValue = new String(decValue);
System.out.println("decryptedValue= " + decryptedValue);
这是我在代码中使用的函数:
public String stringToHex(String base) {
StringBuffer buffer = new StringBuffer();
int intValue = 0;
for (int x = 0; x < base.length(); x++) {
intValue = base.charAt(x);
String hex = Integer.toHexString(intValue);
if (hex.length() == 1) {
buffer.append("0" + hex + "");
} else {
buffer.append(hex + "");
}
}
return buffer.toString();
}
public String byteToAscii(byte[] b, int length) {
String returnString = "";
for (int i = 0; i < length; i++) {
returnString += (char) (b[i] & 0xff);
}
return returnString;
}
我是java加密的新手。请告诉我怎么做? 提前谢谢。
答案 0 :(得分:1)
PKCS5填充字节值必须是填充字节数。例如,如果要填充5个字节,则填充将为05 05 05 05 05
。填充有两个字节,填充为02 02
。
答案 1 :(得分:0)
为什么不试试CipherInputStream和CipherOutputStream?这样你就可以忘记填充。
答案 2 :(得分:0)
temparray[88]
表示您正在访问第89个字节,是不是太多了?
我认为你应该只有:
temparray[87] = 00; // this is byte number 88
然后删除
temparray[88] = 00; // this is byte number 89
答案 3 :(得分:0)
您的错误消息告诉您最后一个块的长度存在问题:“使用填充密码解密时,输入长度必须是8的倍数”
查看输入帧,其长度不是8的倍数。我计算11个完整的8字节块和4个字节的半个块,共计92个字节。在尝试解密之前,您是否应该删除一些未加密的内容?是否有完整的消息?
我还注意到,您一度将缓冲区编码为十六进制,稍后您将其解码为Base-64。那可能不会给你你所期待的。它也可以解释长度问题。十六进制的四个字符是两个字节,而Base-64的四个字符是三个字节。