当前位置:主页 > java教程 > 详解java微信小程序开发中加密解密算法

java微信小程序开发中加密解密算法总结

发布:2019-06-05 09:52:08 201


本站精选了一篇java算法相关的编程文章,网友辛飞昂根据主题投稿了本篇教程内容,涉及到java、微信小程序、加密、解密、算法、详解java微信小程序开发中加密解密算法相关内容,已被120网友关注,下面的电子资料对本篇知识点有更加详尽的解释。

详解java微信小程序开发中加密解密算法

一、概述

微信推出了小程序,很多公司的客户端应用不仅具有了APP、H5、还接入了小程序开发。但是,小程序中竟然没有提供Java版本的加密数据解密算法。这着实让广大的Java开发人员蛋疼。

微信小程序提供的加密数据解密算法链接

我们下载的算法示例如下:

木有Java!! 木有Java!! 木有Java!!

那么如何解决这个问题,我们一起来实现Java版本的微信小程序加密数据解密算法。

二、实现Java版本的微信小程序加密数据解密算法

1、创建项目

这里,我们创建一个Maven工程,具体创建步骤略。

2、配置pom.xml

我们在pom.xml中加入如下配置。

<dependency>
 <groupId>org.bouncycastle</groupId>
 <artifactId>bcprov-jdk16</artifactId>
 <version>1.46</version>
</dependency>
 
 <dependency>
 <groupId>commons-codec</groupId>
 <artifactId>commons-codec</artifactId>
 <version>1.4</version>
</dependency>
 
<dependency> 
 <groupId>net.sf.json-lib</groupId> 
 <artifactId>json-lib</artifactId> 
 <version>2.2.3</version> 
 <classifier>jdk15</classifier> 
</dependency>

3、实现AES类

package com.chwl.medical.crypto.wx;
 
import java.security.AlgorithmParameters;
import java.security.InvalidAlgorithmParameterException;
import java.security.InvalidKeyException;
import java.security.Key;
import java.security.NoSuchAlgorithmException;
import java.security.NoSuchProviderException;
import java.security.Security;
 
import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.IllegalBlockSizeException;
import javax.crypto.NoSuchPaddingException;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import org.bouncycastle.jce.provider.BouncyCastleProvider;
 
/**
 * AES加密
 * @author liuyazhuang
 *
 */
public class AES {
 
 public static boolean initialized = false;
 
 /**
 * AES解密
 * 
 * @param content
 *  密文
 * @return
 * @throws InvalidAlgorithmParameterException
 * @throws NoSuchProviderException
 */
 public byte[] decrypt(byte[] content, byte[] keyByte, byte[] ivByte) throws InvalidAlgorithmParameterException {
 initialize();
 try {
 Cipher cipher = Cipher.getInstance("AES/CBC/PKCS7Padding");
 Key sKeySpec = new SecretKeySpec(keyByte, "AES");
 cipher.init(Cipher.DECRYPT_MODE, sKeySpec, generateIV(ivByte));// 初始化
 byte[] result = cipher.doFinal(content);
 return result;
 } catch (NoSuchAlgorithmException e) {
 e.printStackTrace();
 } catch (NoSuchPaddingException e) {
 e.printStackTrace();
 } catch (InvalidKeyException e) {
 e.printStackTrace();
 } catch (IllegalBlockSizeException e) {
 e.printStackTrace();
 } catch (BadPaddingException e) {
 e.printStackTrace();
 } catch (NoSuchProviderException e) {
 // TODO Auto-generated catch block
 e.printStackTrace();
 } catch (Exception e) {
 // TODO Auto-generated catch block
 e.printStackTrace();
 }
 return null;
 }
 
 public static void initialize() {
 if (initialized)
 return;
 Security.addProvider(new BouncyCastleProvider());
 initialized = true;
 }
 
 // 生成iv
 public static AlgorithmParameters generateIV(byte[] iv) throws Exception {
 AlgorithmParameters params = AlgorithmParameters.getInstance("AES");
 params.init(new IvParameterSpec(iv));
 return params;
 }
}

4、实现WxPKCS7Encoder类

package com.chwl.medical.crypto.wx;
 
import java.nio.charset.Charset;
import java.util.Arrays;
 
 
/**
 * 微信小程序加解密
 * @author liuyazhuang
 *
 */
public class WxPKCS7Encoder {
 private static final Charset CHARSET = Charset.forName("utf-8");
 private static final int BLOCK_SIZE = 32;
 
 /**
 * 获得对明文进行补位填充的字节.
 *
 * @param count
 *  需要进行填充补位操作的明文字节个数
 * @return 补齐用的字节数组
 */
 public static byte[] encode(int count) {
 // 计算需要填充的位数
 int amountToPad = BLOCK_SIZE - (count % BLOCK_SIZE);
 if (amountToPad == 0) {
 amountToPad = BLOCK_SIZE;
 }
 // 获得补位所用的字符
 char padChr = chr(amountToPad);
 String tmp = new String();
 for (int index = 0; index < amountToPad; index++) {
 tmp += padChr;
 }
 return tmp.getBytes(CHARSET);
 }
 
 /**
 * 删除解密后明文的补位字符
 *
 * @param decrypted
 *  解密后的明文
 * @return 删除补位字符后的明文
 */
 public static byte[] decode(byte[] decrypted) {
 int pad = decrypted[decrypted.length - 1];
 if (pad < 1 || pad > 32) {
 pad = 0;
 }
 return Arrays.copyOfRange(decrypted, 0, decrypted.length - pad);
 }
 
 /**
 * 将数字转化成ASCII码对应的字符,用于对明文进行补码
 *
 * @param a
 *  需要转化的数字
 * @return 转化得到的字符
 */
 public static char chr(int a) {
 byte target = (byte) (a & 0xFF);
 return (char) target;
 }
}

5、实现WXCore类

这个类主要是对具体算法的封装,统一对外提供方法。

 

package com.chwl.medical.crypto.wx;
 
import org.apache.commons.codec.binary.Base64;
 
import net.sf.json.JSONObject;
 
 
/**
 * 封装对外访问方法
 * @author liuyazhuang
 *
 */
public class WXCore {
 
 private static final String WATERMARK = "watermark";
 private static final String APPID = "appid";
 /**
 * 解密数据
 * @return
 * @throws Exception
 */
 public static String decrypt(String appId, String encryptedData, String sessionKey, String iv){
 String result = "";
 try {
 AES aes = new AES(); 
 byte[] resultByte = aes.decrypt(Base64.decodeBase64(encryptedData), Base64.decodeBase64(sessionKey), Base64.decodeBase64(iv)); 
 if(null != resultByte && resultByte.length > 0){ 
  result = new String(WxPKCS7Encoder.decode(resultByte)); 
 JSONObject jsonObject = JSONObject.fromObject(result);
 String decryptAppid = jsonObject.getJSONObject(WATERMARK).getString(APPID);
 if(!appId.equals(decryptAppid)){
 result = "";
 }
  } 
 } catch (Exception e) {
 result = "";
 e.printStackTrace();
 }
 return result;
 }
 
 
 public static void main(String[] args) throws Exception{
 String appId = "wx4f4bc4dec97d474b";
 String encryptedData = "CiyLU1Aw2KjvrjMdj8YKliAjtP4gsMZMQmRzooG2xrDcvSnxIMXFufNstNGTyaGS9uT5geRa0W4oTOb1WT7fJlAC+oNPdbB+3hVbJSRgv+4lGOETKUQz6OYStslQ142dNCuabNPGBzlooOmB231qMM85d2/fV6ChevvXvQP8Hkue1poOFtnEtpyxVLW1zAo6/1Xx1COxFvrc2d7UL/lmHInNlxuacJXwu0fjpXfz/YqYzBIBzD6WUfTIF9GRHpOn/Hz7saL8xz+W//FRAUid1OksQaQx4CMs8LOddcQhULW4ucetDf96JcR3g0gfRK4PC7E/r7Z6xNrXd2UIeorGj5Ef7b1pJAYB6Y5anaHqZ9J6nKEBvB4DnNLIVWSgARns/8wR2SiRS7MNACwTyrGvt9ts8p12PKFdlqYTopNHR1Vf7XjfhQlVsAJdNiKdYmYVoKlaRv85IfVunYzO0IKXsyl7JCUjCpoG20f0a04COwfneQAGGwd5oa+T8yO5hzuyDb/XcxxmK01EpqOyuxINew==";
 String sessionKey = "tiihtNczf5v6AKRyjwEUhQ==";
 String iv = "r7BXXKkLb8qrSNn05n0qiA==";
 System.out.println(decrypt(appId, encryptedData, sessionKey, iv));
 }
}


三、测试

1、运行Java版微信小程序加密数据解密算法

这里我们就直接运行WXcore类的main方法,这里的测试数据都是从Python版微信小程序加密数据解密算法的示例程序中提出来的。我们的运行结果如下:

代码如下:
{"openId":"oGZUI0egBJY1zhBYw2KhdUfwVJJE","nickName":"Band","gender":1,"language":"zh_CN","city":"Guangzhou","province":"Guangdong","country":"CN","avatarUrl":"http://wx.qlogo.cn/mmopen/vi_32/aSKcBBPpibyKNicHNTMM0qJVh8Kjgiak2AHWr8MHM4WgMEm7GFhsf8OYrySdbvAMvTsw3mo8ibKicsnfN5pRjl1p8HQ/0","unionId":"ocMvos6NjeKLIBqg5Mr9QjxrP1FA","watermark":{"timestamp":1477314187,"appid":"wx4f4bc4dec97d474b"}}

2、运行Python版微信小程序加密数据解密算法

这里我们在python环境中直接运行微信官方提供的Python版小程序加密数据解密算法,结果如下:

 代码如下:
{u'province': u'Guangdong', u'openId': u'oGZUI0egBJY1zhBYw2KhdUfwVJJE', u'language': u'zh_CN', u'city': u'Guangzhou', u'gender': 1, u'avatarUrl': u'http://wx.qlogo.cn/mmopen/vi_32/aSKcBBPpibyKNicHNTMM0qJVh8Kjgiak2AHWr8MHM4WgMEm7GFhsf8OYrySdbvAMvTsw3mo8ibKicsnfN5pRjl1p8HQ/0', u'watermark': {u'timestamp': 1477314187, u'appid': u'wx4f4bc4dec97d474b'}, u'country': u'CN', u'nickName': u'Band', u'unionId': u'ocMvos6NjeKLIBqg5Mr9QjxrP1FA'}

通过对比以上结果可知,我们自行使用Java实现的Java版微信小程序加密数据解密算法与微信官方提供的Python版小程序加密数据解密算法结果一致。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持码农之家。


参考资料

相关文章

  • JavaScript动态检测密码强度原理及实现方法示例效果

    发布:2020-02-04

    这篇文章主要介绍了JavaScript动态检测密码强度原理及实现方法,结合具体实例形式详细分析了javascript针对输入字符串密码强度检测的原理与相关判断操作技巧,需要的朋友可以参考下


  • Java请求转发和请求重定向区别详解

    发布:2023-01-18

    给大家整理了关于Java的教程,这篇文章主要介绍了Java请求转发和请求重定向区别详解,请求转发和请求重定向,但二者是完全不同的,所以我们今天就来盘他们的区别介绍,需要的朋友可以参考一下


  • JavaScript标准对象知识点总结

    发布:2019-10-01

    这篇文章主要为大家详细介绍了JavaScript标准对象的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下


  • Java程序运行机制及错误分析

    发布:2020-05-21

    这篇文章主要主要介绍了Java虚拟机(JVM)的有关内容以及Java程序的运行机制和错误分析,需要的朋友可以了解下。


  • java 注解的基础详细介绍

    发布:2022-10-10

    给网友朋友们带来一篇关于java的教程,这篇文章主要介绍了java 注解的基础详细介绍的相关资料,希望通过本文大家能掌握注解的使用方法,需要的朋友可以参考下


  • Java中简单实用Quartz概述

    发布:2023-04-01

    Quartz是一个开源的Java调度框架,可以用来实现在指定的时间或时间间隔触发任务执行的功能,这篇文章主要介绍了Java中简单实用Quartz,需要的朋友可以参考下


  • 详解JAVA-4NIO之Channel之间的数据传输方法

    发布:2020-01-22

    下面小编就为大家带来一篇JAVA-4NIO之Channel之间的数据传输方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧


  • Java AQS中CyclicBarrier回环栅栏的使用

    发布:2023-04-15

    这篇文章主要介绍了Java中的 CyclicBarrier详解,CyclicBarrier没有显示继承哪个父类或者实现哪个父接口, 所有AQS和重入锁不是通过继承实现的,而是通过组合实现的,下文相关内容需要的小伙伴可以参考一下


网友讨论