String.getBytes()陷阱

现象

在Eclipse对某数据进行RSA私钥签名(带中文),在Eclipse中运行Tomcat部署的程序中通过RSA公钥对签名进行验证,验证成功;但是独立运行Tomcat再次测试时却验证失败。

分析

  1. 检查密钥对是否匹配,从动态读取改为硬编码,测试结果:仍然失败;
  2. 对签名后数据进行Base64的编码解码,URL编码解码,确保中间传输过程没问题,测试结果:仍然失败;
  3. 尝试调整加密源(纯英文,单个单词,缩小加密字符串长度),纯英文的正常,测试结果:部分成功;
  4. 对比签名数据,发现两种环境下签名数据都不一致,初步怀疑是source.getBytes()有问题,使用如下代码进行测试,在Eclipse里执行时,结果分别是7和12;而在独立命令行执行时结果分别是7和8
    System.out.println("English".getBytes().length);
    System.out.println("中文编码".getBytes().length);

解决

Java的getBytes()应该使用的是操作系统的编码,我的操作系统是简体中文的WIN7,中文的话按2个字节算;而Eclipse里执行时,未带任何环境变量,使用的却是项目的编码UTF-8,中文按3个字节算,导致了最终差异。
通过getBytes("UTF-8")指定编码后,两种环境下测试结果一致,问题解决。


转载于:https://my.oschina.net/u/699015/blog/196492

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
package com.tydic.common.utils; import javax.crypto.Cipher; import javax.crypto.spec.IvParameterSpec; import javax.crypto.spec.SecretKeySpec; import org.apache.commons.codec.binary.Base64; /* * AES加解密算法 * * @author jueyue * 加密用的Key 可以用26个字母和数字组成,最好不要用保留字符,虽然不会错,至于怎么裁决,个人看情况而定 此处使用AES-128-CBC加密模式,key需要为16位。 也是使用0102030405060708 */ public class AES { // 加密 public static String Encrypt(String sSrc, String sKey) throws Exception { if (sKey == null) { System.out.print("Key为空null"); return null; } // 判断Key是否为16位 if (sKey.length() != 16) { System.out.print("Key长度不是16位"); return null; } byte[] raw = sKey.getBytes(); SecretKeySpec skeySpec = new SecretKeySpec(raw, "AES"); Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");//"算法/模式/补码方式" IvParameterSpec iv = new IvParameterSpec("0102030405060708".getBytes());//使用CBC模式,需要一个向量iv,可增加加密算法的强度 cipher.init(Cipher.ENCRYPT_MODE, skeySpec, iv); byte[] encrypted = cipher.doFinal(sSrc.getBytes()); return Base64.encodeBase64String(encrypted);//此处使用BAES64做转码功能,同时能起到2次加密的作用。 } // 解密 public static String Decrypt(String sSrc, String sKey) throws Exception { try { // 判断Key是否正确 if (sKey == null) { System.out.print("Key为空null"); return null; } // 判断Key是否为16位 if (sKey.length() != 16) { System.out.print("Key长度不是16位"); return null; } byte[] raw = sKey.getBytes("ASCII"); SecretKeySpec skeySpec = new SecretKeySpec(raw, "AES"); Cipher ciphe

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值