微信開放平臺(tái)之網(wǎng)站授權(quán)微信登錄功能
1 微信開放平臺(tái):https://open.weixin.qq.com/
3.pc頁面顯示
4. 通過官方提供的文檔,我們可以看出一共分4個(gè)步驟
第一步:請(qǐng)求CODE
第二步:通過code獲取access_token
第三步:通過access_token調(diào)用接口
第4步:獲取用戶個(gè)人信息(UnionID機(jī)制)
api:核心代碼
public class weixin_helper { public weixin_helper() { } /// <summary> /// 根據(jù)AppID和AppSecret獲得access token(默認(rèn)過期時(shí)間為2小時(shí)) /// </summary> /// <returns>Dictionary</returns> public static Dictionary<string, object> get_access_token() { //獲得配置信息 oauth_config config = oauth_helper.get_config(2); string send_url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=" + config.oauth_app_id + "&secret=" + config.oauth_app_key + ""; //發(fā)送并接受返回值 string result = Utils.HttpGet(send_url); if (result.Contains("errmsg")) { return null; } try { Dictionary<string, object> dic = JsonConvert.DeserializeObject<Dictionary<string, object>>(result); return dic; } catch { return null; } } /// <summary> /// 取得臨時(shí)的Access Token(默認(rèn)過期時(shí)間為2小時(shí)) /// </summary> /// <param name="code">臨時(shí)Authorization Code</param> /// <param name="state">防止CSRF攻擊,成功授權(quán)后回調(diào)時(shí)會(huì)原樣帶回</param> /// <returns>Dictionary</returns> public static Dictionary<string, object> get_access_token(string code, string state) { //獲得配置信息 oauth_config config = oauth_helper.get_config(2); string send_url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=" + config.oauth_app_id + "&secret=" + config.oauth_app_key + "&code="+code+"&grant_type=authorization_code"; //發(fā)送并接受返回值 string result = Utils.HttpGet(send_url); if (result.Contains("errmsg")) { return null; } try { Dictionary<string, object> dic = JsonConvert.DeserializeObject<Dictionary<string, object>>(result); return dic; } catch { return null; } } /// <summary> /// 根據(jù)access_token判斷access_token是否過期 /// </summary> /// <param name="access_token"></param> /// <returns>true表示未失效</returns> public static bool check_access_token(string access_token) { //獲得配置信息 oauth_config config = oauth_helper.get_config(2); string send_url = "https://api.weixin.qq.com/sns/auth?access_token=" + access_token + "&openid=" + config.oauth_app_id; //發(fā)送并接受返回值 string result = Utils.HttpGet(send_url); try { Dictionary<string, object> dic = JsonConvert.DeserializeObject<Dictionary<string, object>>(result); if (dic.ContainsKey("errmsg")) { if (dic["errmsg"].ToString()=="ok") { return true; } else { return false; } } return false; } catch { return false; } } /// <summary> /// 若fresh_token已過期則根據(jù)refresh_token取得新的refresh_token /// </summary> /// <param name="refresh_token">refresh_token</param> /// <returns>Dictionary</returns> public static Dictionary<string, object> get_refresh_token(string refresh_token) { //獲得配置信息 oauth_config config = oauth_helper.get_config(2); string send_url = "https://api.weixin.qq.com/sns/oauth2/refresh_token?appid=" + config.oauth_app_id + "&grant_type=refresh_token&refresh_token=" + refresh_token; //發(fā)送并接受返回值 string result = Utils.HttpGet(send_url); if (result.Contains("errmsg")) { return null; } try { Dictionary<string, object> dic = JsonConvert.DeserializeObject<Dictionary<string, object>>(result); return dic; } catch { return null; } } /// <summary> /// 獲取登錄用戶自己的基本資料 /// </summary> /// <param name="access_token">臨時(shí)的Access Token</param> /// <param name="open_id">用戶openid</param> /// <returns>Dictionary</returns> public static Dictionary<string, object> get_user_info(string access_token, string open_id) { //獲得配置信息 oauth_config config = oauth_helper.get_config(2); //發(fā)送并接受返回值 string send_url = "https://api.weixin.qq.com/sns/userinfo?access_token="+access_token+"&openid="+open_id; //發(fā)送并接受返回值 string result = Utils.HttpGet(send_url); if (result.Contains("errmsg")) { return null; } //反序列化JSON Dictionary<string, object> dic = JsonHelper.DataRowFromJSON(result); return dic; } }
控制器的核心代碼:
#region 微信登錄 /// <summary> /// 微信登錄 /// </summary> public ActionResult WeChat() { //獲得配置信息 oauth_config config = oauth_helper.get_config(2); //主鍵id if (config == null) { return Content("出錯(cuò)了,您尚未配置微信相關(guān)的API信息!"); } string state = Guid.NewGuid().ToString().Replace("-", ""); Session["oauth_state"] = state; string send_url = "https://open.weixin.qq.com/connect/qrconnect?appid=" + config.oauth_app_id + "&redirect_uri=" + Utils.UrlEncode(config.return_uri.ToLower()) + "&response_type=code&scope=snsapi_login&state=" + state + "#wechat_redirect"; //開始發(fā)送 return Redirect(send_url); //跳轉(zhuǎn)到微信自己 指定的關(guān)聯(lián)登陸頁面 } /// <summary> /// 微信登錄返回action /// </summary> public ActionResult WeChatReturnUrl(string state, string code) { //取得返回參數(shù) string access_token = string.Empty; string expires_in = string.Empty; string client_id = string.Empty; string openid = string.Empty; string refresh_token = string.Empty; if (Session["oauth_state"] == null || Session["oauth_state"].ToString() == "" || state != Session["oauth_state"].ToString() || string.IsNullOrEmpty(code))//若返回參數(shù)中未包含code或者state沒有通過驗(yàn)證則提示出錯(cuò) { return Content("出錯(cuò)啦,state未初始化!"); } //第一步:通過code來獲取Access Token以及openid Dictionary<string, object> dic1 = weixin_helper.get_access_token(code, state); if (dic1 == null || !dic1.ContainsKey("access_token")) { return Content("錯(cuò)誤代碼:,無法獲取Access Token,請(qǐng)檢查App Key是否正確!"); } if (dic1 == null || !dic1.ContainsKey("openid")) { if (dic1.ContainsKey("errmsg")) { return Content("errcode:" + dic1["errcode"] + ",errmsg:" + dic1["errmsg"]); } else { return Content("出錯(cuò)啦,無法獲取用戶授權(quán)Openid!"); } } access_token = dic1["access_token"].ToString();//獲取access_token expires_in = dic1["expires_in"].ToString();//獲取過期時(shí)間 refresh_token = dic1["refresh_token"].ToString();//獲取用于重新刷新access_token的憑證 openid = dic1["openid"].ToString();//用戶唯一標(biāo)示openid //儲(chǔ)存獲取數(shù)據(jù)用到的信息 Session["oauth_name"] = "webchat"; Session["oauth_access_token"] = access_token; Session["oauth_openid"] = openid; Session["oauth_refresh_token"] = refresh_token; #region todo 將獲取到的用戶信息保存到數(shù)據(jù)庫中 #endregion //第二步:通過Access Token以及openid來獲取用戶的基本信息 //Dictionary<string, object> dic2 = weixin_helper.get_user_info(access_token,openid); //第三步:跳轉(zhuǎn)到指定頁面 return Content(WeChatResultJson()); } /// <summary> /// 微信登錄返回action, 處理用戶信息 /// </summary> public string WeChatResultJson() { string oauth_access_token = string.Empty; string oauth_openid = string.Empty; string oauth_name = string.Empty; string oauth_refresh_token = string.Empty; if (Session["oauth_name"] == null || Session["oauth_access_token"] == null || Session["oauth_openid"] == null) { return "{\"ret\":\"1\", \"msg\":\"出錯(cuò)啦,Access Token已過期或不存在!\"}"; } oauth_name = Session["oauth_name"].ToString(); oauth_access_token = Session["oauth_access_token"].ToString(); oauth_openid = Session["oauth_openid"].ToString(); oauth_refresh_token = Session["oauth_refresh_token"].ToString(); if (!weixin_helper.check_access_token(oauth_access_token)) //調(diào)用access_token前需判斷是否過期 { Dictionary<string, object> dic1 = weixin_helper.get_refresh_token(oauth_refresh_token);//如果已過期則重新?lián)Q取新的access_token if (dic1 == null || !dic1.ContainsKey("access_token")) { return "{\"openid\":\"0\", \"msg\":\"出錯(cuò)啦,無法獲取access_token!\"}"; } oauth_access_token = dic1["access_token"].ToString(); } Dictionary<string, object> dic = weixin_helper.get_user_info(oauth_access_token, oauth_openid); if (dic == null) { return "{\"openid\":\"0\", \"msg\":\"出錯(cuò)啦,無法獲取授權(quán)用戶信息!\"}"; } try { StringBuilder str = new StringBuilder(); str.Append("{"); str.Append("\"openid\": \"" + dic["openid"].ToString() + "\", "); str.Append("\"nickname\": \"" + dic["nickname"].ToString() + "\", "); str.Append("\"sex\": \"" + dic["sex"].ToString() + "\", "); str.Append("\"province\": \"" + dic["province"].ToString() + "\", "); str.Append("\"city\": \"" + dic["city"].ToString() + "\", "); str.Append("\"country\": \"" + dic["country"].ToString() + "\", "); str.Append("\"headimgurl\": \"" + dic["headimgurl"].ToString() + "\", "); str.Append("\"privilege\": \"" + dic["privilege"].ToString() + "\", "); str.Append("\"unionid\": \"" + dic["unionid"].ToString() + "\""); str.Append("\"oauth_name\": \"" + oauth_name + "\""); str.Append("\"oauth_access_token\": \"" + oauth_access_token + "\""); str.Append("\"oauth_openid\": \"" + oauth_openid + "\""); str.Append("}"); return str.ToString(); } catch { return "{\"ret\":\"0\", \"msg\":\"出錯(cuò)啦,無法獲取授權(quán)用戶信息!\"}"; } } #endregion
- 微信公眾平臺(tái)網(wǎng)頁授權(quán)獲取用戶基本信息中授權(quán)回調(diào)域名設(shè)置的變動(dòng)
- 解析微信JS-SDK配置授權(quán),實(shí)現(xiàn)分享接口
- 微信開發(fā) 微信授權(quán)詳解
- 微信公眾號(hào)-獲取用戶信息(網(wǎng)頁授權(quán)獲?。?shí)現(xiàn)步驟
- php版微信公眾平臺(tái)之微信網(wǎng)頁登陸授權(quán)示例
- MVC微信網(wǎng)頁授權(quán)獲取用戶OpenId
- 微信網(wǎng)頁授權(quán)(OAuth2.0) PHP 源碼簡單實(shí)現(xiàn)
- ASP.NET實(shí)現(xiàn)QQ、微信、新浪微博OAuth2.0授權(quán)登錄
- PHP實(shí)現(xiàn)微信網(wǎng)頁授權(quán)開發(fā)教程
- 微信開發(fā)之網(wǎng)頁授權(quán)獲取用戶信息(二)
- weiphp微信公眾平臺(tái)授權(quán)設(shè)置
- 解決微信授權(quán)回調(diào)頁面域名只能設(shè)置一個(gè)的問題
相關(guān)文章
C#中LINQ多條件JOIN時(shí)為什么可以使用匿名類
這篇文章主要給大家介紹了關(guān)于C#中LINQ多條件JOIN時(shí)為什么可以使用匿名類的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面來一起看看吧2018-09-09WPF實(shí)現(xiàn)繪制統(tǒng)計(jì)圖(柱狀圖)的方法詳解
這篇文章主要為大家詳細(xì)介紹了如何基于WPF實(shí)現(xiàn)實(shí)現(xiàn)統(tǒng)計(jì)圖(柱狀圖)的繪制,文中的示例代碼簡潔易懂,對(duì)我們學(xué)習(xí)WPF有一定幫助,感興趣的可以了解一下2022-07-07C#事務(wù)處理(Execute Transaction)實(shí)例解析
這篇文章主要介紹了C#事務(wù)處理(Execute Transaction)實(shí)例解析,對(duì)于理解和學(xué)習(xí)事務(wù)處理有一定的幫助,需要的朋友可以參考下2014-08-08基于WPF簡單實(shí)現(xiàn)Meesage消息提醒
這篇文章主要介紹了如何利用WPF簡單實(shí)現(xiàn)Meesage消息提醒,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)或工作有一定幫助,需要的可以參考一下2023-07-07C# Newtonsoft.Json 解析多嵌套json 進(jìn)行反序列化的實(shí)例
這篇文章主要介紹了C# Newtonsoft.Json 解析多嵌套json 進(jìn)行反序列化的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2021-01-01C#中的自動(dòng)類型轉(zhuǎn)換和強(qiáng)制類型轉(zhuǎn)換
這篇文章主要介紹了C#中的自動(dòng)類型轉(zhuǎn)換和強(qiáng)制類型轉(zhuǎn)換,非常不錯(cuò),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2019-08-08