前言

在小程序的项目中,对小程序授权获取的code值进行解析是一个必要的操作,因为需要获取到其中的openid、session_key、unionid进行一个身份的验证。判断该用户是新用户或是其它操作(具体根据自己的业务需求修改就行) 其中unionid需要主体账户绑定了小程序后才可以获取到。所以在没有绑定的时候,我们解析code是只能获取到openid和session_key两个参数的

接下来我们开始操作一下,代码和详细操作如下👇

一、Java解析微信小程序code,获取openid、session_key、unionid

1、先写一个code解析的工具类

WechatUtils

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
// vo实体类参数
import com.entityVo.TestEntityVO ;
import com.utils.TestUtils;

import org.bouncycastle.util.encoders.Base64;
import lombok.extern.slf4j.Slf4j;
import org.apache.http.util.TextUtils;
import org.bouncycastle.jce.provider.BouncyCastleProvider;
import org.springframework.stereotype.Component;

import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.io.*;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLConnection;
import java.security.AlgorithmParameters;
import java.security.Security;
import java.util.*;

@Slf4j(topic = "WechatUtils")
@Component
public class WechatUtil {
	private static final String appId = "*****";
	private static final String secret = "*x**x****";

	/**
	 * 获取小程序code换取openid、session_key
	 *
	 * @param code
	 * @return
	 */
	public static JSONObject getOpenId(String code) {

		String url = "https://api.weixin.qq.com/sns/jscode2session?appid=" + appId
				+ "&secret=" + secret + "&js_code=" + code + "&grant_type=authorization_code";
		PrintWriter out = null;
		BufferedReader in = null;
		String line;
		StringBuffer stringBuffer = new StringBuffer();
		try {
			URL realUrl = new URL(url);
			// 打开和URL之间的连接
			URLConnection conn = realUrl.openConnection();

			// 设置通用的请求属性 设置请求格式
			//设置返回类型
			conn.setRequestProperty("contentType", "text/plain");
			//设置请求类型
			conn.setRequestProperty("content-type", "application/x-www-form-urlencoded");
			//设置超时时间
			conn.setConnectTimeout(1000);
			conn.setReadTimeout(1000);
			conn.setDoOutput(true);
			conn.connect();
			// 获取URLConnection对象对应的输出流
			out = new PrintWriter(conn.getOutputStream());
			// flush输出流的缓冲
			out.flush();
			// 定义BufferedReader输入流来读取URL的响应    设置接收格式
			in = new BufferedReader(
					new InputStreamReader(conn.getInputStream(), "UTF-8"));
			while ((line = in.readLine()) != null) {
				stringBuffer.append(line);
			}
			JSONObject jsonObject = JSONObject.parseObject(stringBuffer.toString());
			return jsonObject;

		} catch (Exception e) {
			e.printStackTrace();
		}
		//使用finally块来关闭输出流、输入流
		finally {
			try {
				if (out != null) {
					out.close();
				}
				if (in != null) {
					in.close();
				}
			} catch (IOException ex) {
				ex.printStackTrace();
			}
		}
		return null;
	}
}

2、业务逻辑层的业务代码

(具体实体类自己根据自己的业务需求配置就行)
service

public interface WxOperationService{
	TestEntity getCodeInfo(TestEntityVO vo);
}

impl实现类

@Service
pulic class WxOperationServiceImpl implements WxOperationService{
	
	@Override
	public TestEntity getCodeInfo(TestEntityVO vo){
		TestEntity entity = new TestEntity;
		// 获取code数据
		JSONObject object = WechatUtils.getOpenId(vo.getCode())
		// json数据转换成字符串
		String openid = wx.get("openid").toString();
		String sessionkey = wx.get("session_key").toString();
		// 当主体账户绑定小程序后就可以获取到,未绑定无法获取
        String unionId="";
        if(!EmptyUtils.isEmpty(wx.get("unionid"))){
            unionId=wx.get("unionid").toString();
        }
        entity.setOpenId(openid);
        entity.setSessionKey(sessionkey);
        entity.setUnionId(unionid);
        // 是否需要获取手机号,true需要
        entity.setIsFlagPhone(true);
        System.out.println("openid:"+openid+",session_key:"+sessionkey+",unionid:"+unionid);
        // 返回参数
		return entity;
	}
}

👆这个代码已经可以解析出code信息,具体的业务流程自己去添加就ok


二、Java获取微信用户手机号

在上面的代码中我们已经获取到了openid和session_key了,而code信息中是不能获取到用户的手机号码的,解析微信手机号的相关数据需要openid和session_key才行,所以在上面代码中,我将相关数据返回给了前端,前端将我传回的参数,还有第二次请求中的iv、encryptedData一起传回给后端,然后我们进行解密

接下来我们操作一下,代码如下👇

1、工具类中添加解密方法

还是我们之前的WechatUtils,在里面添加getPhone方法

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
// vo实体类参数
import com.entity.TestEntityVO;
import com.utils.TestUtils;

import org.bouncycastle.util.encoders.Base64;
import lombok.extern.slf4j.Slf4j;
import org.apache.http.util.TextUtils;
import org.bouncycastle.jce.provider.BouncyCastleProvider;
import org.springframework.stereotype.Component;

import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.io.*;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLConnection;
import java.security.AlgorithmParameters;
import java.security.Security;
import java.util.*;

@Slf4j(topic = "WechatUtils")
@Component
public class WechatUtil {
	private static final String appId = "*****";
	private static final String secret = "********";

	/**
	 * 获取小程序code换取openid、session_key
	 *
	 * @param code
	 * @return
	 */
	public static JSONObject getOpenId(String code) {

		String url = "https://api.weixin.qq.com/sns/jscode2session?appid=" + appId
				+ "&secret=" + secret + "&js_code=" + code + "&grant_type=authorization_code";
		PrintWriter out = null;
		BufferedReader in = null;
		String line;
		StringBuffer stringBuffer = new StringBuffer();
		try {
			URL realUrl = new URL(url);
			// 打开和URL之间的连接
			URLConnection conn = realUrl.openConnection();

			// 设置通用的请求属性 设置请求格式
			//设置返回类型
			conn.setRequestProperty("contentType", "text/plain");
			//设置请求类型
			conn.setRequestProperty("content-type", "application/x-www-form-urlencoded");
			//设置超时时间
			conn.setConnectTimeout(1000);
			conn.setReadTimeout(1000);
			conn.setDoOutput(true);
			conn.connect();
			// 获取URLConnection对象对应的输出流
			out = new PrintWriter(conn.getOutputStream());
			// flush输出流的缓冲
			out.flush();
			// 定义BufferedReader输入流来读取URL的响应    设置接收格式
			in = new BufferedReader(
					new InputStreamReader(conn.getInputStream(), "UTF-8"));
			while ((line = in.readLine()) != null) {
				stringBuffer.append(line);
			}
			JSONObject jsonObject = JSONObject.parseObject(stringBuffer.toString());
			return jsonObject;

		} catch (Exception e) {
			e.printStackTrace();
		}
		//使用finally块来关闭输出流、输入流
		finally {
			try {
				if (out != null) {
					out.close();
				}
				if (in != null) {
					in.close();
				}
			} catch (IOException ex) {
				ex.printStackTrace();
			}
		}
		return null;
	}
}

public static Map<String, Object> getPhoneNumber(TestEntityVO vo) {
		Map<String,Object> map=new HashMap<>();
		String openid= vo.getWechatOpenId();
		String session_key = vo.getSessionKey();
		if (!EmptyUtils.isEmpty(openid)) {

			if(EmptyUtils.isEmpty(session_key)){
				return null;
			}
			map.put("openid",openid);
			// 被加密的数据
			byte[] dataByte = Base64.decode(vo.getEncryptedData());
			// 加密秘钥
			byte[] keyByte = Base64.decode(session_key);
			// 偏移量
			byte[] ivByte = Base64.decode(vo.getIv());
			try {
				// 如果密钥不足16位,那么就补足. 这个if 中的内容很重要
				int base = 16;
				String result = null;
				if (keyByte.length % base != 0) {
					int groups = keyByte.length / base + (keyByte.length % base != 0 ? 1 : 0);
					byte[] temp = new byte[groups * base];
					Arrays.fill(temp, (byte) 0);
					System.arraycopy(keyByte, 0, temp, 0, keyByte.length);
					keyByte = temp;
				}
				// 初始化
				Security.addProvider(new BouncyCastleProvider());
				Cipher cipher = Cipher.getInstance("AES/CBC/PKCS7Padding", "BC");
				SecretKeySpec spec = new SecretKeySpec(keyByte, "AES");
				AlgorithmParameters parameters = AlgorithmParameters.getInstance("AES");
				parameters.init(new IvParameterSpec(ivByte));
				// 初始化
				cipher.init(Cipher.DECRYPT_MODE, spec, parameters);
				byte[] resultByte = cipher.doFinal(dataByte);
				if (null != resultByte && resultByte.length > 0) {
					result = new String(resultByte, "UTF-8");
					JSONObject jsonObject = JSONObject.parseObject(result);
					map.put("param",jsonObject);
					return map;
				}
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
		return null;
	}

2、业务逻辑层代码

service

public interface WxOperationService{
	TestEntity getCodeInfo(TestEntityVO vo);
	
	TestEntityDTO getCodeInfo(TestEntityVO vo);
}

impl实现类

@Override
	public TestEntityDTO getCodeInfo(TestEntityVO vo) throws CoBusinessException {
	TestEntityDTO dto = new TestEntityDTO ;
	Map<String, Object> map = WechatUtil.getPhoneNumber(param);
		if (EmptyUtils.isEmpty(map)) {
			throw new CoBusinessException(CoReturnFormat.WECHAT_ERROR);
		}
		String phone = "";
		Object phoneNumber = map.get("param");
		String jsonString = JSONObject.toJSONString(phoneNumber);
		JSONObject obj = JSONObject.parseObject(jsonString);
		if (!EmptyUtils.isEmpty(jsonString)) {
			phone = obj.get("phoneNumber").toString();
		}
		dto.setPhone(phone);
		return dto;
}

👆以上就是本次的笔记了,大家有需要用的自取,有用记得点个赞噢