1. 第三方发起微信授权登录请求,微信用户允许授权第三方应用后,微信会拉起应用或重定向到第三方网站,并且带上授权临时票据code参数;

2. 通过code参数加上AppID和AppSecret等,通过API换取access_token;

3. 通过access_token进行接口调用,获取用户基本数据资源或帮助用户实现基本操作。

获取access_token时序图:

微信网站应用授权登录接口对接代码实现:

需用的配置文件:

需用的vo类:

代码:

package com.thinkwin.common.vo;

/**

* 类名: WechatAccessTokenVo  * 描述: 授权关系接口的调用凭证模型 * 开发人员: weining  * 创建时间:  2017/5/5  */

public class WechatAccessTokenVo{

// 网页授权接口调用凭证

private String accessToken ;

// 凭证有效时长

private int expiresIn;

// 用于刷新凭证

private String refreshToken;

// 用户标识

private String openId;

// 用户授权作用域

private String scope;

// 用户全局唯一标识  unionid

private String unionid;

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 getRefreshToken() {

return refreshToken;

}

public void setRefreshToken(String refreshToken) {

this.refreshToken = refreshToken;

}

public String getOpenId() {

return openId;

}

public void setOpenId(String openId) {

this.openId = openId;

}

public String getScope() {

return scope;

}

public void setScope(String scope) {

this.scope = scope;

}

public String getUnionid() {

return unionid;

}

public void setUnionid(String unionid) {

this.unionid = unionid;

}

}

package com.thinkwin.common.vo;

import java.util.List;

/**

* 类名: WechatSNSUserInfoVo  * 描述: 通过网页授权获取的用户信息  * 开发人员: weining  * 创建时间:  2017/4/27  */

public class WechatSNSUserInfoVo {

// 用户标识

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 privilegeList;

// 用户全局唯一标识  unionid

private String unionid;

public String getOpenId() {

return openId;

}

public void setOpenId(String openId) {

this.openId = openId;

}

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 getHeadImgUrl() {

return headImgUrl;

}

public void setHeadImgUrl(String headImgUrl) {

this.headImgUrl = headImgUrl;

}

public List getPrivilegeList() {

return privilegeList;

}

public void setPrivilegeList(List privilegeList) {

this.privilegeList = privilegeList;

}

public String getUnionid() {

return unionid;

}

public void setUnionid(String unionid) {

this.unionid = unionid;

}

}

需要的工具类:

代码 :

package com.thinkwin.common.utils.wechat;

import com.alibaba.fastjson.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;

/**

* 类名: WechatCommonUtil  * 描述: 微信登录通用工具类 * 开发人员: weining  * 创建时间:  2017/5/5  */

public class WechatCommonUtil {

private static Logger log = LoggerFactory.getLogger(WechatCommonUtil.class);

/**

* 发送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 WechatX509TrustManager() };

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.parseObject(buffer.toString());

} catch (ConnectException ce) {

log.error("连接超时:{}", ce);

} catch (Exception e) {

log.error("https请求异常:{}", e);

}

return jsonObject;

}

}

package com.thinkwin.common.utils.wechat;

import org.slf4j.Logger;

import org.slf4j.LoggerFactory;

import java.io.IOException;

import java.io.InputStream;

import java.util.Properties;

/**

* 类名: WechatConfigLoader  * 描述: 微信初始化配置信息 * 开发人员: weining  * 创建时间:  2017/5/5  */

public class WechatConfigLoader {

//日志记录对象

private static Logger log = LoggerFactory.getLogger(WechatConfigLoader.class);

//配置文件路径

private static String wechatPath = "wechat.properties";

//开发平台应用唯一标识

private static String appId;

//开放平台应用密钥

private static String appSecret;

//微信第三方回调地址

private static String backUrl;

static {

// 类初始化后加载配置文件

InputStream in = WechatConfigLoader.class.getClassLoader()

.getResourceAsStream(wechatPath);

Properties props = new Properties();

try {

props.load(in);

} catch (IOException e) {

log.error("load wechat setting error,pleace check the file path:"

+ wechatPath);

log.error(e.toString(), e);

}

appId = props.getProperty("wechat.appId");

appSecret = props.getProperty("wechat.appSecret");

backUrl = props.getProperty("wechat.backUrl");

log.debug("load wechat setting success,file path:" + wechatPath);

}

public static String getAppId() {

return appId;

}

public static String getAppSecret() {

return appSecret;

}

public static String getBackUrl() {

return backUrl;

}

public static void setWechatPath(String wechatPath) {

WechatConfigLoader.wechatPath = wechatPath;

}

public static String getWechatPath() {

return wechatPath;

}

public static void setAppId(String appId) {

WechatConfigLoader.appId = appId;

}

public static void setAppSecret(String appSecret) {

WechatConfigLoader.appSecret = appSecret;

}

public static void setBackUrl(String backUrl) {

WechatConfigLoader.backUrl = backUrl;

}

}

package com.thinkwin.lehui.common.wechatUtil;

import com.alibaba.fastjson.JSONArray;

import com.alibaba.fastjson.JSONObject;

import com.thinkwin.lehui.common.wechatVo.WechatAccessTokenVo;

import com.thinkwin.lehui.common.wechatVo.WechatSNSUserInfoVo;

import org.slf4j.Logger;

import org.slf4j.LoggerFactory;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import java.io.IOException;

import java.net.URLEncoder;

import java.util.List;

/**

* 类名: WXOAuthProcess  * 描述: 微信第三方登录授权流程工具类 * 开发人员: weining  * 创建时间:  2017/5/5  */

public class WechatOAuthProcessUtil {

private static Logger log = LoggerFactory.getLogger(WechatOAuthProcessUtil.class);

/**

* 1.获取授权code

* @param req

* @param resp

*/

public static void getOAuthCode(HttpServletRequest req, HttpServletResponse resp){

String appId = WechatConfigLoader.getAppId();

String backUrl = WechatConfigLoader.getBackUrl();

String url="https://open.weixin.qq.com/connect/qrconnect?appid=APPID&redirect_uri=REDIRECT_URI&response_type=code&scope=snsapi_login&state=STATE#wechat_redirect";

url = url.replace("APPID",appId);

url = url.replace("REDIRECT_URI",URLEncoder.encode(backUrl));

try {

resp.sendRedirect(url);

} catch (IOException e) {

e.printStackTrace();

}

}

/**

* 2.获取授权调用token

* @param appId       开发平台应用唯一标识

* @param appSecret   开放平台应用密钥

* @param code        授权临时票据 根据code来换取accessToken

*/

public static WechatAccessTokenVo getOauthAccessToken(String appId, String appSecret, String code){

WechatAccessTokenVo wechatAccessTokenVo = null;

//拼接微信获取accessToken请求的链接

String url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=APPID&secret=SECRET&code=CODE&grant_type=authorization_code";

url = url.replace("APPID",appId);

url = url.replace("SECRET",appSecret);

url = url.replace("CODE",code);

// 获取网页授权凭证 发送https请求

JSONObject jsonObject = (WechatCommonUtil.httpsRequest(url, "GET", null));

if (null != jsonObject) {

try {

wechatAccessTokenVo = new WechatAccessTokenVo();

wechatAccessTokenVo.setAccessToken(jsonObject.getString("access_token"));

wechatAccessTokenVo.setExpiresIn(jsonObject.getInteger("expires_in"));

wechatAccessTokenVo.setRefreshToken(jsonObject.getString("refresh_token"));

wechatAccessTokenVo.setOpenId(jsonObject.getString("openid"));

wechatAccessTokenVo.setScope(jsonObject.getString("scope"));

wechatAccessTokenVo.setUnionid(jsonObject.getString("unionid"));

} catch (Exception e) {

wechatAccessTokenVo = null;

int errorCode = jsonObject.getInteger("errcode");

String errorMsg = jsonObject.getString("errmsg");

log.error("获取网页授权凭证失败 errcode:{} errmsg:{}", errorCode, errorMsg);

}

}

return wechatAccessTokenVo;

}

/**

* 3.通过网页授权获取用户信息

*

* @param accessToken 网页授权接口调用凭证

* @param openId 用户标识

* @return SNSUserInfo

*/

@SuppressWarnings( { "deprecation", "unchecked" })

public static WechatSNSUserInfoVo getSNSUserInfo(String accessToken, String openId) {

WechatSNSUserInfoVo snsUserInfo = null;

// 拼接请求地址 发送https请求

String url = "https://api.weixin.qq.com/sns/userinfo?access_token=ACCESS_TOKEN&openid=OPENID";

url = url.replace("ACCESS_TOKEN", accessToken);

url = url.replace("OPENID", openId);

// 通过网页授权获取用户信息

JSONObject jsonObject = WechatCommonUtil.httpsRequest(url, "GET", null);

if (null != jsonObject) {

try {

snsUserInfo = new WechatSNSUserInfoVo();

// 用户的标识

snsUserInfo.setOpenId(jsonObject.getString("openid"));

// 昵称

snsUserInfo.setNickname(jsonObject.getString("nickname"));

// 性别(1是男性,2是女性,0是未知)

snsUserInfo.setSex(jsonObject.getInteger("sex"));

// 用户所在国家

snsUserInfo.setCountry(jsonObject.getString("country"));

// 用户所在省份

snsUserInfo.setProvince(jsonObject.getString("province"));

// 用户所在城市

snsUserInfo.setCity(jsonObject.getString("city"));

// 用户头像

snsUserInfo.setHeadImgUrl(jsonObject.getString("headimgurl"));

// 用户特权信息

snsUserInfo.setPrivilegeList(JSONArray.toJavaObject(jsonObject.getJSONArray("privilege"), List.class));

} catch (Exception e) {

snsUserInfo = null;

int errorCode = jsonObject.getInteger("errcode");

String errorMsg = jsonObject.getString("errmsg");

log.error("获取用户信息失败 errcode:{} errmsg:{}", errorCode, errorMsg);

}

}

return snsUserInfo;

}

/**

* 刷新授权调用token

* @param appId       开发平台应用唯一标识

* @param refreshToken    通过access_token获取到的refresh_token参数

*/

public static WechatAccessTokenVo refreshAccessToken(String appId, String refreshToken){

WechatAccessTokenVo  wechatAccessTokenVo = null;

//拼接微信刷新accessToken请求的链接

String url = "https://api.weixin.qq.com/sns/oauth2/refresh_token?appid=APPID&grant_type=refresh_token&refresh_token=REFRESH_TOKEN";

url = url.replace("APPID",appId);

url = url.replace("REFRESH_TOKEN",refreshToken);

// 获取网页授权凭证 发送https请求

JSONObject jsonObject = WechatCommonUtil.httpsRequest(url, "GET", null);

if (null != jsonObject) {

try {

wechatAccessTokenVo = new WechatAccessTokenVo();

wechatAccessTokenVo.setAccessToken(jsonObject.getString("access_token"));

wechatAccessTokenVo.setExpiresIn(jsonObject.getInteger("expires_in"));

wechatAccessTokenVo.setRefreshToken(jsonObject.getString("refresh_token"));

wechatAccessTokenVo.setOpenId(jsonObject.getString("openid"));

wechatAccessTokenVo.setScope(jsonObject.getString("scope"));

} catch (Exception e) {

wechatAccessTokenVo = null;

int errorCode = jsonObject.getInteger("errcode");

String errorMsg = jsonObject.getString("errmsg");

log.error("获取网页授权凭证失败 errcode:{} errmsg:{}", errorCode, errorMsg);

}

}

return wechatAccessTokenVo;

}

}

package com.thinkwin.common.utils.wechat;

import javax.net.ssl.X509TrustManager;

import java.security.cert.CertificateException;

import java.security.cert.X509Certificate;

/**

* 类名: MyX509TrustManager  * 描述: 信任管理器   * 开发人员: weining  * 创建时间:  2017/5/5  */

public class WechatX509TrustManager implements X509TrustManager {

// 检查客户端证书

public void checkClientTrusted(X509Certificate[] x509Certificates, String s) throws CertificateException {

}

// 检查服务器端证书

public void checkServerTrusted(X509Certificate[] x509Certificates, String s) throws CertificateException {

}

// 返回受信任的X509证书数组

public X509Certificate[] getAcceptedIssuers() {

return new X509Certificate[0];

}

}

最后就是controller控制层代码:

/**

* 确认请求来自微信服务器  微信的回调

*/

@RequestMapping(value = "/oauthtest", method = RequestMethod.GET)

public String OAuthTest(HttpServletRequest request, HttpServletResponse response, Model model) throws ServletException, IOException {

request.setCharacterEncoding("utf-8");

response.setCharacterEncoding("utf-8");

PrintWriter out = response.getWriter();

// 用户同意授权后,能获取到code

String code = request.getParameter("code");

String state = request.getParameter("state");

// 用户同意授权

if (!"authdeny".equals(code)) {

// 获取网页授权access_token

WechatAccessTokenVo wechatAccessTokenVo = WechatOAuthProcessUtil.getOauthAccessToken(code);

if(null == wechatAccessTokenVo){

if(state.equals("register")){

return "redirect:/system/wechatregisterpage";

}

return "redirect:/system/loginpage";

}

// 网页授权接口访问凭证

String accessToken = wechatAccessTokenVo.getAccessToken();

// 用户标识

String openId = wechatAccessTokenVo.getOpenId();

// 获取用户信息

WechatSNSUserInfoVo snsUserInfo = WechatOAuthProcessUtil.getSNSUserInfo(accessToken, openId);

Map map = new HashMap<>();

map.put("wechatAccessTokenVo",wechatAccessTokenVo);

map.put("snsUserInfo",snsUserInfo);

String s = JSON.toJSONString(map);

// 用户unionid

String unionId = snsUserInfo.getUnionid();

//把字符串存redis里面

RedisUtil.set("WeChat"+unionId,s);

//访问数据库操作

//直接登录

return "登录页";

}

return null;

}

到这里java基于微信开放平台的授权登录功能就完成了。

微信第三方扫描登录 java源代码_微信开放平台基于网站应用授权登录源码(java)...相关推荐

  1. JAVA毕业设计web家教信息服务平台设计与实现计算机源码+lw文档+系统+调试部署+数据库

    JAVA毕业设计web家教信息服务平台设计与实现计算机源码+lw文档+系统+调试部署+数据库 JAVA毕业设计web家教信息服务平台设计与实现计算机源码+lw文档+系统+调试部署+数据库 本源码技术栈 ...

  2. 基于JAVA社区养老综合服务平台服务端计算机毕业设计源码+数据库+lw文档+系统+部署

    基于JAVA社区养老综合服务平台服务端计算机毕业设计源码+数据库+lw文档+系统+部署 基于JAVA社区养老综合服务平台服务端计算机毕业设计源码+数据库+lw文档+系统+部署 本源码技术栈: 项目架构 ...

  3. java计算机毕业设计汽车租赁平台的设计与实现源码+数据库+lw文档+系统

    java计算机毕业设计汽车租赁平台的设计与实现源码+数据库+lw文档+系统 java计算机毕业设计汽车租赁平台的设计与实现源码+数据库+lw文档+系统 本源码技术栈: 项目架构:B/S架构 开发语言: ...

  4. 计算机毕业设计Java的创意项目众筹平台的设计与开发(源码+系统+mysql数据库+lw文档)

    计算机毕业设计Java的创意项目众筹平台的设计与开发(源码+系统+mysql数据库+lw文档) 计算机毕业设计Java的创意项目众筹平台的设计与开发(源码+系统+mysql数据库+lw文档) 本源码技 ...

  5. java计算机毕业设计烟草经销平台的设计与实现源码+系统+lw+数据库+调试运行

    java计算机毕业设计烟草经销平台的设计与实现源码+系统+lw+数据库+调试运行 注意:该项目只展示部分功能,如需了解,文末获取源码地址. 临近学期结束,还是毕业设计,你还在做java程序.网络编程. ...

  6. java计算机毕业设计汽车租赁平台的设计与实现源码+数据库+系统+lw文档+mybatis+运行部署

    java计算机毕业设计汽车租赁平台的设计与实现源码+数据库+系统+lw文档+mybatis+运行部署 java计算机毕业设计汽车租赁平台的设计与实现源码+数据库+系统+lw文档+mybatis+运行部 ...

  7. ssm+java农村快递代取平台52wxh(程序+lw+源码+远程部署)

    项目运行 项目含有源码(见文末).文档.程序.数据库.配套开发软件.软件安装教程 环境配置: Jdk1.8 + Tomcat7.0 + Mysql + HBuilderX(Webstorm也行)+ E ...

  8. ssm+java计算机毕业设计拼车平台u5398(程序+lw+源码+远程部署)

    项目运行 项目含有源码(见文末).文档.程序.数据库.配套开发软件.软件安装教程 环境配置: Jdk1.8 + Tomcat7.0 + Mysql + HBuilderX(Webstorm也行)+ E ...

  9. 吾爱破解java核心技术_这才是某音解析最新源码,别拿第三方的来糊弄人

    [PHP] 纯文本查看 复制代码<?php header("Content-type: text/html; charset=utf-8"); $url_g=@$_GET[' ...

最新文章

  1. python使用np.argsort对一维numpy概率值数据排序获取倒序索引、获取的top索引(例如top2、top5、top10)索引二维numpy数组中对应的原始数据:原始数据概率最大的头部数据
  2. 某书2018面试题之最长对称子串
  3. fstream,sstream,使用(习题8.16)
  4. Android开发各类常见错误解决方案
  5. ubuntu16.04安装UR3/UR5/UR10机械臂的ROS驱动并实现gazebo下Moveit运动规划仿真以及真实UR3机械臂的运动控制(3)
  6. 你不得不掌握的thinkphp5
  7. [渝粤教育] 西南科技大学 英美文学 在线考试复习资料
  8. NanShan即时通讯 ie6 ie7 ie8 不支持json 终极解放方案
  9. 十年维护专家的“秘籍
  10. hunnu11543:小明的烦恼——分糖果
  11. 剪贴板(下)-剪贴板查看器
  12. 至少连接一个aura sync兼容设备_电磁兼容测试照片
  13. 私有云部署和本地化部署有什么区别?
  14. Python编程 | 统计新浪微博热门话题
  15. java 多音词语转拼音_py4j汉语转拼音多音字处理
  16. 梦想近在眼前却遥不可及
  17. X280 工业4g路由器
  18. python软件是免费的吗-python免费吗
  19. 【研究日记】虚拟歌姬自动调教之歌曲音频切割的问题
  20. R8-1 中序和后序创建二叉树

热门文章

  1. Why Redis 4.0?
  2. 彻底理解内存泄漏,memory leak
  3. 飞桨企业版重磅发布智能边缘控制台 5分钟零代码自动化模型部署
  4. 年初新立Flag,新华三解决方案部做了点儿啥?
  5. OpenStack精华问答 | OpenStack与计算虚拟化有什么关系?
  6. 英特尔助力金山云带你畅游云端的游戏世界
  7. 清华计算机系和交叉信息学院,清华大学交叉信息研究院与计算机科学与技术系哪个..._考研_帮考网...
  8. 预登录握手失败_英雄联盟手游登录问题汇总
  9. Docker Nacos Mysql集群
  10. Linux Shell脚本专栏_一键查看服务器的利用率_04