java获取微信openId和用户信息

小程序获取登录code和用户加密信息方法
      wx.login({
        success(res) {
          console.log("postLogin login success", res);
          if (!res.code) {
            reject();
          }
          wx.getUserInfo({
            withCredentials: true,
            success(userRes) {
              app.api.user.postLogin({
                'code': res.code,
                "signature":userRes.signature,
                "rawData":userRes.rawData,
                'encryptedData': userRes.encryptedData,
                'iv': userRes.iv
              }).onSuccess(function (res) {
                if (res) {
                  resolve(res);
                } else {
                  reject();
                  util.showErrorMessage("登录失败");
                }
              }).onFail(function (res) {
                reject();
                util.showErrorMessage(res.msg);
              }).start();
            }
          })
        }
      });
小程序传参对象类WxLoginVO,代码注释已经说明相关参数,这里就不做解释
package com.frank.lmsg.vo.req;

import java.io.Serializable;

/**
 *
 * Description: 用户信息 微信登录验证VO
 * Auth: Frank
 * Date: 2019-08-09
 * Time: 下午 2:22
 */
public class WxLoginVO implements Serializable {
    /**
     * 对应小程序获取用户信息的encryptedData字段
     */
    private String encryptedData;

    /**
     * 对应小程序获取用户信息的iv字段
     */
    private String iv;

    /**
     * 对应小程序获取用户信息的rawData字段
     */
    private String rawData;

    /**
     * 对应小程序获取用户信息的signature字段
     */
    private String signature;

    /**
     * 对应小程序登录的时候反的code字段
     */
    private String code;

    public String getEncryptedData() {
        return encryptedData;
    }

    public void setEncryptedData(String encryptedData) {
        this.encryptedData = encryptedData;
    }

    public String getIv() {
        return iv;
    }

    public void setIv(String iv) {
        this.iv = iv;
    }

    public String getRawData() {
        return rawData;
    }

    public void setRawData(String rawData) {
        this.rawData = rawData;
    }

    public String getSignature() {
        return signature;
    }

    public void setSignature(String signature) {
        this.signature = signature;
    }

    public String getCode() {
        return code;
    }

    public void setCode(String code) {
        this.code = code;
    }

    @Override
    public String toString() {
        return "WxLoginVO{" +
                "encryptedData='" + encryptedData + '\'' +
                ", iv='" + iv + '\'' +
                ", rawData='" + rawData + '\'' +
                ", signature='" + signature + '\'' +
                ", code='" + code + '\'' +
                '}';
    }
}


微信相关工具类WxUtils,代码注释已经说明相关方法,这里就不做解释
package com.frank.lmsg.base.utils;

import com.alibaba.fastjson.JSONObject;
import com.frank.lmsg.constant.WxConsts;
import com.frank.lmsg.vo.resp.UserInfoVo;
import com.frank.lmsg.vo.resp.WxUserInfoVO;
import org.apache.commons.codec.digest.DigestUtils;
import org.apache.logging.log4j.LogManager;
import org.bouncycastle.jce.provider.BouncyCastleProvider;

import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import java.nio.charset.StandardCharsets;
import java.security.AlgorithmParameters;
import java.security.Security;
import java.text.MessageFormat;
import java.util.Arrays;
import java.util.Base64;

/**
 * 微信用户信息解密工具
 *
 * @author Giam
 * @version 1.0
 * @date 2019/6/19
 */
public class WxUtils {
    /**
     * 微信获取用户信息 encryptedData解密
     * <p>
     *     解密算法如下:
     *     1.对称解密使用的算法为 AES-128-CBC,数据采用PKCS#7填充。
     *     2. 对称解密的目标密文为 Base64_Decode(encryptedData)。
     *     3. 对称解密秘钥 aeskey = Base64_Decode(session_key), aeskey 是16字节。
     *     4. 对称解密算法初始向量 为Base64_Decode(iv),其中iv由数据接口返回。
     * </p>
     *
     * @param encryptedData 加密的数据
     * @param sessionKey sessionKey
     * @param iv iv
     * @return
     */
    public static WxUserInfoVO encryptedDataUserInfo(String encryptedData, String sessionKey, String iv){
        // 被加密的数据
        Base64.Decoder decoder = Base64.getDecoder();

        byte[] dataByte = decoder.decode(encryptedData);
        // 加密秘钥
        byte[] keyByte = decoder.decode(sessionKey);
        // 偏移量
        byte[] ivByte = decoder.decode(iv);

        try {
            // 如果密钥不足16位,那么就补足.  这个if 中的内容很重要
            int base = 16;
            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) {
                String result = new String(resultByte, StandardCharsets.UTF_8);
                return JSONObject.parseObject(result, WxUserInfoVO.class);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    /**
     *  获得微信signature
     *  signature = sha1( rawData + session_key )
     * @param rawData rawData
     * @param sessionKey sessionKey
     * @return
     */
    public static String getSignature(String rawData, String sessionKey) {
        StringBuffer signatureStr = new StringBuffer();
        signatureStr.append(rawData);
        signatureStr.append(sessionKey);
        return DigestUtils.sha1Hex(signatureStr.toString().getBytes());
    }

    /**
     * 获取登录信息URL并拼接相关参数
     * @param api
     * @param arg
     * @return
     */
    public static String getApi(String api, Object... arg) {
        String url = String.format(api, WxConsts.APP_ID, WxConsts.APP_SECRET);
        url = MessageFormat.format(url, arg);
        return url;
    }


}

登录微信请求工具类HttpUtils,代码注释已经说明相关方法,这里就不做解释

package com.frank.lmsg.base.utils;

import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.utils.URIBuilder;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.util.EntityUtils;

import java.io.IOException;
import java.net.URI;
import java.util.Map;

public class HttpUtils {
    public static String doGet(String url, Map<String, String> param) {

        // 创建Httpclient对象
        CloseableHttpClient httpclient = HttpClients.createDefault();

        String resultString = "";
        CloseableHttpResponse response = null;
        try {
            // 创建uri
            URIBuilder builder = new URIBuilder(url);
            if (param != null) {
                for (String key : param.keySet()) {
                    builder.addParameter(key, param.get(key));
                }
            }
            URI uri = builder.build();

            // 创建http GET请求
            HttpGet httpGet = new HttpGet(uri);

            // 执行请求
            response = httpclient.execute(httpGet);
            // 判断返回状态是否为200
            if (response.getStatusLine().getStatusCode() == 200) {
                resultString = EntityUtils.toString(response.getEntity(), "UTF-8");
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                if (response != null) {
                    response.close();
                }
                httpclient.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        return resultString;
    }

    public static String doGet(String url) {
        return doGet(url, null);
    }
}

详细代码可以查看:
https://github.com/fuxingkai/LMsg
https://github.com/fuxingkai/LMsg-B
参考地址:
https://developers.weixin.qq.com/miniprogram/dev/api/open-api/login/wx.login.html
https://developers.weixin.qq.com/miniprogram/dev/api-backend/open-api/login/auth.code2Session.html
https://developers.weixin.qq.com/miniprogram/dev/api/open-api/user-info/wx.getUserInfo.html
https://developers.weixin.qq.com/miniprogram/dev/framework/open-ability/signature.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,928评论 6 509
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,748评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,282评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,065评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,101评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,855评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,521评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,414评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,931评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,053评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,191评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,873评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,529评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,074评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,188评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,491评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,173评论 2 357

推荐阅读更多精彩内容