java实现微信授权登录

服务端实现app授权登录

1.导入jar包

<!--后台查询微信用户信息-->
<dependency>
     <groupId>com.github.liyiorg</groupId>
     <artifactId>weixin-popular</artifactId>
     <version>2.8.5</version>
 </dependency>

2.工具类

package com.es.biz.common.utils;

import com.es.biz.modules.user.entity.wechat.WechatTokenEntity;
import com.es.biz.modules.user.entity.wechat.WechatUserEntity;
import net.sf.json.JSONException;
import net.sf.json.JSONObject;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.net.ssl.HttpsURLConnection;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLSocketFactory;
import javax.net.ssl.TrustManager;
import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.net.ConnectException;
import java.net.URL;
import java.nio.charset.StandardCharsets;

public class WechatUtils {
    private final static Logger log = LoggerFactory.getLogger(WechatUtils.class);
    private final static String appid = "xxxxxxxx"; //凭证
    private final static String appsecret = "xxxxxxxxx"; //凭证密钥

    // 凭证获取(GET)
    public final static String tokenUrl = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=APPID&secret=SECRET&code=CODE&grant_type=authorization_code";
    //userinfo
    public final static String userInfoUrl = "https://api.weixin.qq.com/sns/userinfo?access_token=ACCESS_TOKEN&openid=OPENID&lang=zh_CN";

    /**
     * 获取接口访问凭证
     *
     * @param code app授权后传回
     * @return
     */
    public static WechatTokenEntity getToken(String code) {
        WechatTokenEntity token = new WechatTokenEntity();
//        long now = new Date().getTime();
//        if (tokenTime != 0 && now - tokenTime < 7000000) {//token有效时间 7e6 毫秒
//            return token;
//        }
        String requestUrl = tokenUrl.replace("APPID", appid).replace("SECRET", appsecret).replace("CODE", code);
        // 发起GET请求获取凭证
        JSONObject jsonObject = httpsRequest(requestUrl, "GET", null);
        if (null != jsonObject) {
            try {
                token.setUnionId(jsonObject.getString("unionid"));
                token.setOpenid(jsonObject.getString("openid"));
                token.setAccessToken(jsonObject.getString("access_token"));
                token.setRefreshToken(jsonObject.getString("refresh_token"));
                token.setExpiresIn(jsonObject.getInt("expires_in"));
            } catch (JSONException e) {
                token = null;
                // 获取token失败
                log.error("获取token失败 errcode:{} errmsg:{}", jsonObject.getInt("errcode"), jsonObject.getString("errmsg"));
            }
        }
        return token;
    }

    public static void main(String args[]) {
        // 获取接口访问凭证
        String accessToken = getToken("填入你的授权code").getAccessToken();
        String openid = getToken("填入你的授权code").getOpenid();
        /**
         * 获取用户信息
         */
        WechatUserEntity user = getUserInfo(accessToken, openid);
        //做这个测试的时候可以先关注,或者取消关注,控制台会打印出来此用户的openid
        System.out.println("OpenID:" + user.getOpenId());
        System.out.println("关注状态:" + user.getSubscribe());
        System.out.println("关注时间:" + user.getSubscribeTime());
        System.out.println("昵称:" + user.getNickname());
        System.out.println("性别:" + user.getSex());
        System.out.println("国家:" + user.getCountry());
        System.out.println("省份:" + user.getProvince());
        System.out.println("城市:" + user.getCity());
        System.out.println("语言:" + user.getLanguage());
        System.out.println("头像:" + user.getHeadImgUrl());
    }

    /**
     * 获取用户信息
     *
     * @param accessToken 接口访问凭证
     * @param openId      用户标识
     * @return WeixinUserInfo
     */
    public static WechatUserEntity getUserInfo(String accessToken, String openId) {

        WechatUserEntity wechatUserEntity = null;
        // 拼接请求地址
        String requestUrl = userInfoUrl.replace("ACCESS_TOKEN", accessToken).replace("OPENID", openId);
        // 获取用户信息
        JSONObject jsonObject = httpsRequest(requestUrl, "GET", null);
        if (null != jsonObject) {
            try {
                wechatUserEntity = new WechatUserEntity();
                // 用户的标识
                wechatUserEntity.setOpenId(jsonObject.getString("openid"));
                wechatUserEntity.setUnionId(jsonObject.getString("unionid"));
                // 关注状态(1是关注,0是未关注),未关注时获取不到其余信息
//                wechatUserEntity.setSubscribe(jsonObject.getInt("subscribe"));
                // 用户关注时间
//                wechatUserEntity.setSubscribeTime(jsonObject.getString("subscribe_time"));
                // 昵称
                wechatUserEntity.setNickname(jsonObject.getString("nickname"));
                // 用户的性别(1是男性,2是女性,0是未知)
//                wechatUserEntity.setSex(jsonObject.getInt("sex"));
                // 用户所在国家
//                wechatUserEntity.setCountry(jsonObject.getString("country"));
                // 用户所在省份
//                wechatUserEntity.setProvince(jsonObject.getString("province"));
                // 用户所在城市
//                wechatUserEntity.setCity(jsonObject.getString("city"));
                // 用户的语言,简体中文为zh_CN
                wechatUserEntity.setLanguage(jsonObject.getString("language"));
                // 用户头像
                wechatUserEntity.setHeadImgUrl(jsonObject.getString("headimgurl"));
            } catch (Exception e) {
                int errorCode = jsonObject.getInt("errcode");
                String errorMsg = jsonObject.getString("errmsg");
//                System.err.printf("获取用户信息失败 errcode:{} errmsg:{}", errorCode, errorMsg);
            }
        }
        return wechatUserEntity;
    }

    /**
     * 发送https请求
     *
     * @param requestUrl    请求地址
     * @param requestMethod 请求方式(GET、POST)
     * @param outputStr     提交的数据
     * @return JSONObject(通过JSONObject.get ( key)的方式获取json对象的属性值)
     */
    public static JSONObject httpsRequest(String requestUrl, String requestMethod, String outputStr) {
        JSONObject jsonObject = null;
        try {
            // 创建SSLContext对象,并使用我们指定的信任管理器初始化
            TrustManager[] tm = {new WechatTrustManager()};
            SSLContext sslContext = SSLContext.getInstance("SSL", "SunJSSE");
            sslContext.init(null, tm, new java.security.SecureRandom());
            // 从上述SSLContext对象中得到SSLSocketFactory对象
            SSLSocketFactory ssf = sslContext.getSocketFactory();

            URL url = new URL(requestUrl);
            HttpsURLConnection conn = (HttpsURLConnection) url.openConnection();
            conn.setSSLSocketFactory(ssf);

            conn.setDoOutput(true);
            conn.setDoInput(true);
            conn.setUseCaches(false);
            // 设置请求方式(GET/POST)
            conn.setRequestMethod(requestMethod);

            // 当outputStr不为null时向输出流写数据
            if (null != outputStr) {
                OutputStream outputStream = conn.getOutputStream();
                // 注意编码格式
                outputStream.write(outputStr.getBytes(StandardCharsets.UTF_8));
                outputStream.close();
            }

            // 从输入流读取返回内容
            InputStream inputStream = conn.getInputStream();
            InputStreamReader inputStreamReader = new InputStreamReader(inputStream, StandardCharsets.UTF_8);
            BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
            String str = null;
            StringBuilder stringBuilder = new StringBuilder();
            while ((str = bufferedReader.readLine()) != null) {
                stringBuilder.append(str);
            }

            // 释放资源
            bufferedReader.close();
            inputStreamReader.close();
            inputStream.close();
            inputStream = null;
            conn.disconnect();
            jsonObject = JSONObject.fromObject(stringBuilder.toString());
        } catch (ConnectException ce) {
            System.err.printf("连接超时:{}", ce);
        } catch (Exception e) {
            System.err.printf("https请求异常:{}", e);
        }
        return jsonObject;
    }
}


3.信任管理

package com.es.biz.common.utils;

import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;
import javax.net.ssl.X509TrustManager;

/**
 * 描述:信任管理器 </br>
 * 发布版本:V1.0 </br>
 */
/*
 * 证书管理器的作用是让它新人我们指定的证书,
 * 此类中的代码意味着信任所有的证书,不管是不是权威机构颁发的。
 */
public class WechatTrustManager implements X509TrustManager {
    // 检查客户端证书
    public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {
    }

    // 检查服务器端证书
    public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
    }

    // 返回受信任的X509证书数组
    public X509Certificate[] getAcceptedIssuers() {
        return null;
    }
}

4.微信用户详情实体类

package com.es.biz.modules.user.entity.wechat;

import java.io.Serializable;

/**
 * 用户管理InfoVO
 *
 * @author es
 * @email es@126.com
 */
public class WechatUserEntity implements Serializable {
    private static final long serialVersionUID = 1L;

    /**
     * 用户的标识
     */
    private String openId;
    /**
     * 关注状态(1是关注,0是未关注),未关注时获取不到其余信息
     */
    private int subscribe;
    /**
     * 用户关注时间,为时间戳。如果用户曾多次关注,则取最后关注时间
     */
    private String subscribeTime;
    /**
     * 昵称
     */
    private String nickname;
    /**
     * 用户的性别(1是男性,2是女性,0是未知)
     */
    private int sex;
    /**
     * 用户所在国家
     */
    private String country;
    /**
     * 用户所在省份
     */
    private String province;
    /**
     * 用户所在城市
     */
    private String city;
    /**
     * 用户的语言,简体中文为zh_CN
     */
    private String language;
    /**
     * 用户头像
     */
    private String headImgUrl;
    /**
     * 用户特权信息
     */
    private String PrivilegeList;
    /**
     * 微信授权用户唯一标识
     */
    private String unionId;

    public String getOpenId() {
        return openId;
    }

    public void setOpenId(String openId) {
        this.openId = openId;
    }

    public int getSubscribe() {
        return subscribe;
    }

    public void setSubscribe(int subscribe) {
        this.subscribe = subscribe;
    }

    public String getSubscribeTime() {
        return subscribeTime;
    }

    public void setSubscribeTime(String subscribeTime) {
        this.subscribeTime = subscribeTime;
    }

    public String getNickname() {
        return nickname;
    }

    public void setNickname(String nickname) {
        this.nickname = nickname;
    }

    public int getSex() {
        return sex;
    }

    public void setSex(int sex) {
        this.sex = sex;
    }

    public String getCountry() {
        return country;
    }

    public void setCountry(String country) {
        this.country = country;
    }

    public String getProvince() {
        return province;
    }

    public void setProvince(String province) {
        this.province = province;
    }

    public String getCity() {
        return city;
    }

    public void setCity(String city) {
        this.city = city;
    }

    public String getLanguage() {
        return language;
    }

    public void setLanguage(String language) {
        this.language = language;
    }

    public String getHeadImgUrl() {
        return headImgUrl;
    }

    public void setHeadImgUrl(String headImgUrl) {
        this.headImgUrl = headImgUrl;
    }

    public String getPrivilegeList() {
        return PrivilegeList;
    }

    public void setPrivilegeList(String privilegeList) {
        PrivilegeList = privilegeList;
    }

    public String getUnionId() {
        return unionId;
    }

    public void setUnionId(String unionId) {
        this.unionId = unionId;
    }
}

5.微信token实体类

package com.es.biz.modules.user.entity.wechat;

/**
 * 描述: 凭证 </br>
 * 发布版本:V1.0 </br>
 */
public class WechatTokenEntity {
    /**
     * 接口访问凭证֤
     */
    private String accessToken;
    /**
     * 接口访问凭证֤,刷新
     */
    private String refreshToken;
    /**
     * 凭证有效期单位:second
     */
    private int expiresIn;
    /**
     * 授权用户唯一标识
     */
    private String openid;
    /**
     * 微信用户唯一标识
     */
    private String unionId;

    public String getOpenid() {
        return openid;
    }

    public void setOpenid(String openid) {
        this.openid = openid;
    }

    public String getAccessToken() {
        return accessToken;
    }

    public void setAccessToken(String accessToken) {
        this.accessToken = accessToken;
    }

    public int getExpiresIn() {
        return expiresIn;
    }

    public void setExpiresIn(int expiresIn) {
        this.expiresIn = expiresIn;
    }

    public String getUnionId() {
        return unionId;
    }

    public void setUnionId(String unionId) {
        this.unionId = unionId;
    }

    public String getRefreshToken() {
        return refreshToken;
    }

    public void setRefreshToken(String refreshToken) {
        this.refreshToken = refreshToken;
    }
}

评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值