首页 > 其他分享 >微信授权登录:移动端[unionid](一)

微信授权登录:移动端[unionid](一)

时间:2022-10-19 21:33:54浏览次数:70  
标签:jsonObject return String 登录 微信 param unionid null public



如果你有多端登录统一用户,或者是同一产品下不同子产品之间统一用户的需求的话,请提前在微信开放平台打通微信网页授权并绑定,不然的话后期打通用户处理起来比较麻烦。因为同一微信用户不同公众号生成的openid不一样,这样没办法统一用户,在微信开放平台绑定后,会有一个会有unionid,这个unionid是唯一的,这就可以统一用户,当然要花300RMB ,没有这种需求的话,可以不用管。

一,准备工作

①,微信公众平台申请配置,详情见微信分享开发:准备工作[微信公众平台以及微信中控服务配置](一)

②,微信公众平台已经给了详细的授权登录流程接口,我们只需要根据其提供的文档按照步骤实现即可,下面我将根据微信公众平台提供的文档结合项目进行实现。​​微信公众平台​

微信授权登录:移动端[unionid](一)_json


微信授权登录:移动端[unionid](一)_json_02

编辑

二,开发实现预热

①,获取code

https://open.weixin.qq.com/connect/oauth2/authorize?appid=APPID&redirect_uri=REDIRECT_URI&response_type=code&scope=SCOPE&state=STATE#wechat_redirect


微信授权登录:移动端[unionid](一)_网页授权_03

备注:redirect_uri需要进行urlEncode编码,scope使用的是snsapi_userinfo,下面提供一个范例

​https://open.weixin.qq.com/connect/oauth2/authorize?appid=wx010f4709f3abffe3&redirect_uri=http%3A%2F%2Fdevelope.sh01net.com%2Fauth%2FloginByWeixin&response_type=code&scope=snsapi_userinfo#wechat_redirect​

唤起微信授权页面

微信授权登录:移动端[unionid](一)_微信_04


微信授权登录:移动端[unionid](一)_微信_05

编辑

确认登录后会换取到一个code值,并回调到redirect_uri配置的方法函数上并带着code值。

②,通过code换取access_token和openid,然后通过openid换取用户信息并保存。

③,简单的流程图

微信授权登录:移动端[unionid](一)_网页授权_06


微信授权登录:移动端[unionid](一)_网页授权_07

编辑

登录成功页面自由跳转实现方式:实现进入他人分享的指定页面:[带登录权限机制]

三,代码实现(精简了一下,仅供参考,不一定适用你的业务需求,但业务流程都一样的)

①,Controller层(微信回调地址)

@RequestMapping(value = "/loginByWeixin",method = {RequestMethod.GET,RequestMethod.POST})
public String LoginByWeixin(HttpServletResponse response, String code) throws IOException{
System.out.print("======code======"+code);
if(code==null || "".equals(code.trim())){
response.sendRedirect("http://xxxxx/xc-ui-pc-static-portal/index.html#/login");//我这里跳转的是页面,也可以跳转相应方法
return null;
}
try {
String token;//weixin openId
String accessToken;//weixin accessToken
String unionid;
String APPID="";//在微信公众平台申请的appid
String APP_SECRET="";//在微信公众平台申请的APPSECRET
WeixinOauth2Token weixinOauth2Token =AdvancedUtil.getOauth2AccessToken(APPID, APP_SECRET, code);
// weixinOauth2Token 微信请求可能会报错。报错后为null,避免循环
if(weixinOauth2Token==null || weixinOauth2Token.getOpenId()==null){
response.sendRedirect("http://develope.sh01net.com/xc-ui-pc-static-portal/index.html#/login");
return null;
}
token = weixinOauth2Token.getOpenId();
accessToken = weixinOauth2Token.getAccessToken();
//获取微信用户信息
SNSUserInfo snsUserInfo = AdvancedUtil.getSNSUserInfo(accessToken, token);
//查询当前微信是否登录过,你可以根据openid查询或者Unionid查询,具体根据什么查询根据自己的实际情况
unionid=snsUserInfo.getUnionid();
XcUser xcUser =authService.getUserByUnionid(unionid);//替换成自己的查询方法
String access_token=null;

if(xcUser == null ){

//存储用户信息到数据库。比较简单,具体代码省略...
}
response.sendRedirect("http://xxxxx/xc-ui-pc-static-portal/index.html#/search");//这里跳转授权登录成功的页面或方法
return null;
} catch (IOException e) {
e.printStackTrace();
logger.error(e);
response.sendRedirect("http://xxxxx/xc-ui-pc-static-portal/index.html");
return null;
}
}


微信授权登录:移动端[unionid](一)_网页授权_08

②WeixinOauth2Token.java

/**
* @author
public class WeixinOauth2Token {
/**网页授权接口调用凭证 */
private String accessToken;
/**凭证有效时长 */
private int expiresIn;
/**用于刷新凭证 */
private String refreshToken;
/**用户标识 */
private String openId;
/**用户授权作用域 */
private String scope;
/**
* 返回 网页授权接口调用凭证
*
* @return
public String getAccessToken() {
return accessToken;
}
/**
* 设置 网页授权接口调用凭证
*
* @param

public void setAccessToken(String accessToken) {
this.accessToken = accessToken;
}
/**
* 返回 凭证有效时长
*
* @return
public int getExpiresIn() {
return expiresIn;
}
/**
* 设置 凭证有效时长
*
* @param

public void setExpiresIn(int {
this.expiresIn = expiresIn;
}
/**
* 返回 用于刷新凭证
*
* @return
public String getRefreshToken() {
return refreshToken;
}
/**
* 设置 用于刷新凭证
*
* @param

public void setRefreshToken(String refreshToken) {
this.refreshToken = refreshToken;
}
/**
* 返回 用户标识
*
* @return
public String getOpenId() {
return openId;
}
/**
* 设置 用户标识
*
* @param

public void setOpenId(String openId) {
this.openId = openId;
}
/**
* 返回 用户授权作用域
*
* @return
public String getScope() {
return scope;
}
/**
* 设置 用户授权作用域
*
* @param

public void setScope(String scope) {
this.scope = scope;
}

}


微信授权登录:移动端[unionid](一)_json_09

③,SNSUserInfo.java

/**
* @author
public class SNSUserInfo {
/**用户标识 */
private String openId;
/**用户昵称 */
private String nickname;
/**性别(1是男性,2是女性,0是未知) */
private int sex;
/**国家 */
private String country;
/**省份 */
private String province;
/**城市 */
private String city;
/**用户头像链接 */
private String headImgUrl;
/**用户特权信息 */
private List<String> privilegeList;
/**用户unionid,需要绑定开放平台才会获取到 */
private String unionid;

/**
* 返回 用户标识
*
* @return
public String getOpenId() {
return openId;
}
/**
* 设置 用户标识
*
* @param

public void setOpenId(String openId) {
this.openId = openId;
}
/**
* 返回 用户昵称
*
* @return
public String getNickname() {
return nickname;
}
/**
* 设置 用户昵称
*
* @param

public void setNickname(String nickname) {
this.nickname = nickname;
}
/**
* 返回 性别(1是男性,2是女性,0是未知)
*
* @return
public int getSex() {
return sex;
}
/**
* 设置 性别(1是男性,2是女性,0是未知)
*
* @param

public void setSex(int {
this.sex = sex;
}
/**
* 返回 国家
*
* @return
public String getCountry() {
return country;
}
/**
* 设置 国家
*
* @param

public void setCountry(String country) {
this.country = country;
}
/**
* 返回 省份
*
* @return
public String getProvince() {
return province;
}
/**
* 设置 省份
*
* @param

public void setProvince(String province) {
this.province = province;
}
/**
* 返回 城市
*
* @return
public String getCity() {
return city;
}
/**
* 设置 城市
*
* @param

public void setCity(String city) {
this.city = city;
}
/**
* 返回 用户头像链接
*
* @return
public String getHeadImgUrl() {
return headImgUrl;
}
/**
* 设置 用户头像链接
*
* @param

public void setHeadImgUrl(String headImgUrl) {
this.headImgUrl = headImgUrl;
}
/**
* 返回 用户特权信息
*
* @return
public List<String> getPrivilegeList() {
return privilegeList;
}
/**
* 设置 用户特权信息
*
* @param

public void setPrivilegeList(List<String> privilegeList) {
this.privilegeList = privilegeList;
}

/**
* 返回 用户Unionid
*
* @return
public String getUnionid() {
return unionid;
}

/**
* 设置用户Unionid
*
* @param
public void setUnionid(String unionid) {
this.unionid = unionid;
}
}


微信授权登录:移动端[unionid](一)_json_10

④,AdvancedUtil .java 

/**
* @author
public class AdvancedUtil {
private static final Logger logger = LogManager.getLogger(AdvancedUtil.class);
/**
* 获取网页授权凭证
*
* @param appId
* 公众账号的唯一标识
* @param appSecret
* 公众账号的密钥
* @param code
* @return
public static WeixinOauth2Token getOauth2AccessToken(String appId,
String appSecret, String code) {
WeixinOauth2Token wat = null;
// 拼接请求地址
String requestUrl = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=APPID&secret=SECRET&code=CODE&grant_type=authorization_code";
requestUrl = requestUrl.replace("APPID", appId);
requestUrl = requestUrl.replace("SECRET", appSecret);
requestUrl = requestUrl.replace("CODE", code);
// 获取网页授权凭证
JSONObject jsonObject = CommonUtil.httpsRequest(requestUrl, "GET", null);
logger.info("// 获取网页授权凭证access_token"+jsonObject);
if (null != jsonObject) {
try {
wat = new WeixinOauth2Token();
wat.setAccessToken(jsonObject.getString("access_token"));
wat.setExpiresIn(jsonObject.getInt("expires_in"));
wat.setRefreshToken(jsonObject.getString("refresh_token"));
wat.setOpenId(jsonObject.getString("openid"));
wat.setScope(jsonObject.getString("scope"));
} catch (Exception e) {
wat = null;
int errorCode = jsonObject.getInt("errcode");
String errorMsg = jsonObject.getString("errmsg");
logger.info("获取网页授权凭证失败 errcode:{} errmsg:{}" + errorCode + errorMsg);
e.printStackTrace();
}
}

return wat;
}


/**
* 通过网页授权获取用户信息
*
* @param accessToken
* 网页授权接口调用凭证
* @param openId
* 用户标识
* @return
@SuppressWarnings({ "deprecation", "unchecked" })
public static SNSUserInfo getSNSUserInfo(String accessToken, String openId) {
SNSUserInfo snsUserInfo = null;
// 拼接请求地址
String requestUrl = "https://api.weixin.qq.com/sns/userinfo?access_token=ACCESS_TOKEN&openId=OPENID&lang=zh_CN";
requestUrl = requestUrl.replace("ACCESS_TOKEN", accessToken).replace(
"OPENID", openId);
// 通过网页授权获取用户信息
JSONObject jsonObject = CommonUtil.httpsRequest(requestUrl, "GET", null);
logger.info("// 通过网页授权获取用户信息"+jsonObject);
if (null != jsonObject) {
try {
snsUserInfo = new SNSUserInfo();
// 用户的标识
snsUserInfo.setOpenId(openId);
// 昵称
snsUserInfo.setNickname(jsonObject.getString("nickname"));
// 性别(1是男性,2是女性,0是未知)
snsUserInfo.setSex(jsonObject.getInt("sex"));
// 用户所在国家
snsUserInfo.setCountry(jsonObject.getString("country"));
// 用户所在省份
snsUserInfo.setProvince(jsonObject.getString("province"));
// 用户所在城市
snsUserInfo.setCity(jsonObject.getString("city"));
// 用户头像
snsUserInfo.setHeadImgUrl(jsonObject.getString("headimgurl"));
// 用户特权信息
snsUserInfo.setPrivilegeList(JSONArray.toList(
jsonObject.getJSONArray("privilege"), List.class));
//获取unionid
snsUserInfo.setUnionid(jsonObject.getString("unionid"));
} catch (Exception e) {
int errorCode = jsonObject.getInt("errcode");
String errorMsg = jsonObject.getString("errmsg");
logger.info("获取用户信息失败 errcode:{} errmsg:{}" + errorCode + errorMsg);
e.printStackTrace();
}
}

return


微信授权登录:移动端[unionid](一)_json_11

⑤,CommonUtil.java

/**
* 发送https请求
*
* @param requestUrl
* 请求地址
* @param requestMethod
* 请求方式(GET、POST)
* @param outputStr
* 提交的数据
* @return
/**
* 发送https请求
*
* @param requestUrl 请求地址
* @param requestMethod 请求方式(GET、POST)
* @param outputStr 提交的数据
* @return
public static JSONObject httpsRequest(String requestUrl, String requestMethod, String outputStr) {
JSONObject jsonObject = null;
try {
// 创建SSLContext对象,并使用我们指定的信任管理器初始化
TrustManager[] tm = { new MyX509TrustManager() };
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("UTF-8"));
outputStream.close();
}

// 从输入流读取返回内容
InputStream inputStream = conn.getInputStream();
InputStreamReader inputStreamReader = new InputStreamReader(inputStream, "utf-8");
BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
String str = null;
StringBuffer buffer = new StringBuffer();
while ((str = bufferedReader.readLine()) != null) {
buffer.append(str);
}

// 释放资源
bufferedReader.close();
inputStreamReader.close();
inputStream.close();
inputStream = null;
conn.disconnect();
jsonObject = JSONObject.fromObject(buffer.toString());
} catch (ConnectException ce) {

ce.printStackTrace();
} catch (Exception e) {

e.printStackTrace();
}
return


微信授权登录:移动端[unionid](一)_网页授权_12

⑥MyX509TrustManager.java(发送https请求绕过认证,注意引入的包)

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

/**
* @author
public class MyX509TrustManager 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;
}
}


微信授权登录:移动端[unionid](一)_微信_13

⑦,至此,已经实现了微信授权登录的业务。


标签:jsonObject,return,String,登录,微信,param,unionid,null,public
From: https://blog.51cto.com/lvyq/5768530

相关文章

  • 极光手机号码一键登录(二、集成 客户端&服务端)
      Android建议采用mavenCentral自动集成方式;iOS采用cocopods自动集成的方式 文档地址:https://docs.jiguang.cn/jverification/guideline/intro......
  • Uniapp 验证没登录跳转
    逻辑:我随便写一个没有cookie我就跳转到登录页,而且任何返回都没用1.在App.vue里写onload判断是否存在Cookie【有效否后面验证】   2.在导入Uniapp拦截器......
  • 解决公众号网页 微信免登录 重定向 地址 只能带一个参数问题
    场景描述两个手机交互涉及公司内容就没有截图了通过二维码扫码进入判断扫码微信还是支付宝界面,判断之后微信免登陆进行重定向带值传递到后台交互界面1.生成二维码界......
  • 微信开发工具自定义扩展插件
    1、Vscode安装Easy-less2、扩展文件夹用everything搜索mrcrowl-easy-less3、复制出来放桌面,然后通过以下步骤用微信开发工具导入扩展插件就OK了4、输出wxss文......
  • 登录(cookie,session)
    cookie和登录校验cookie:存储在浏览器的一段字符串,最大5kb,每个域都有一个cookie,跨域不共享,格式如k1=v1;k2=v2(可结构化)cookie会随http请求传递给服务端每次http请求,coo......
  • Microsoft Teams团队频道里共享文件弹出登录提示解决办法
    MicrosoftTeams团队频道里共享文件弹出登录提示解决办法一,情况员工从Teams里的团队频道打开共享文件时,该文件突然要求员工登录账号,即使登录账号后依然会弹出下一个,提示......
  • .net core 微信支付-----申请退款
     首先看官方文档:      请求示例:  返回参数:  这个参数太长了就不多说了,上代码:Model:publicclassApplyAmount{///<summary......
  • 查询每个用户最后一次登录信息
    文章目录表结构和数据:方法一:如果只需要用户id和最后一次登陆时间方法二:如果需要携带其他日志信息方法三:oracle数据库使用row_num()over表结构和数据:CREATETABLE`temp_t......
  • 网页在线客服代码-侧边悬浮在线客服/QQ/微信/电话代码
    什么是网页在线客服代码?在线客服系统是通过独立应用程序或嵌入式软件进行的近实时消息交换。早期互联网发展还不流行的时候,那时候的网页结构还比较单一,很多企业的网站上只......
  • RHCE第四章 SSH远程登录服务器
    连接加密技术简介     ssh基于tcp协议  端口号:22目前常见的网络数据包加密技术通常是通过“非对称密钥系统”来处理的。主要通过两把不一样的公钥与私钥来进......