微信授权步骤与详解 -- c#篇
微信授权步骤与详解 -- c#篇
注:这里不涉及界面操作,只介绍代码操作。
1.基本原理如下:

从图上所知,第一步用户访问我们的网页,第二步我们后台跳转到微信授权页面,第三步用户点击授权,第四步微信重定向到第三方(我们后台)并且返回code,第五步请求accesstoken获取accesstoken和openid。
2.详细介绍
第一步,用户访问我们网页。
例如,http://test.authorization.com/Main/TA
第二步,我们后台跳转到微信授权页面。(第一次握手)
首先,我们跳转到微信页面,传参数为appid、redirectUrl、response_type、scope、state。
url为:"https://open.weixin.qq.com/connect/oauth2/authorize?appid=" + appId + "&redirect_uri=" + redirectUrl + "&response_type=code&scope=snsapi_userinfo&state=" + state + "#wechat_redirect"
redirectUrl 为当前后台请求的url。代码如下:
string response_type = "code";// 返回类型,此时固定为:code
string scope = "snsapi_userinfo";//应用授权作用域,snsapi_base (不弹出授权页面,直接跳转,只能获取用户openid),snsapi_userinfo (弹出授权页面,可通过openid拿到昵称、性别、所在地。并且,即使在未关注的情况下,只要用户授权,也能获取其信息)
string state = "state";//随便写,微信提供给我们的自定义参数。重定向后会带上state参数,开发者可以填写a-zA-Z0-9的参数值,最多128字节
string url = string.Format("https://open.weixin.qq.com/connect/oauth2/authorize?appid={0}&redirect_uri={1}&response_type={2}&scope={3}&state={4}#wechat_redirect",appid, redirect_uri, response_type, scope, state);
hconHttpContextBase.Response.Redirect(url, true);//跳转到微信授权
第三步,用户点击授权。

第四步,微信重定向到第三方(我们后台)并且返回code(第二次握手)
微信重定向回我们后台时,会返回code,所以我们可以判断是否有code来确定微信是第几次握手。没code是第一次握手,有code是第二次握手。
第五步,请求accesstoken获取accesstoken和openid
请求代码如下:
string url = "https://api.weixin.qq.com/sns/oauth2/access_token";//请求的url
string sUrlpara = "appid=" + appId + "&secret=" + appSecret +
"&code=" + code + "&grant_type=authorization_code";//请求的参数
string jsonStr = Toos.HttpGet(url, sUrlpara);//网络get请求
//放回json解析
WeiXinAccessTokenResult result = new WeiXinAccessTokenResult();
if (jsonStr.Contains("errcode"))
{
WeiXinErrorMsg errorResult = new WeiXinErrorMsg();
errorResult = JsonHelper.ParseFormJson<WeiXinErrorMsg>(jsonStr);
result.ErrorResult = errorResult;
result.Result = false;
}
else
{
WeiXinAccessTokenModel model = new WeiXinAccessTokenModel();
model = JsonHelper.ParseFormJson<WeiXinAccessTokenModel>(jsonStr);
result.SuccessResult = model;
result.Result = true;
}
返回的json是:
//正确的时候:
{
"access_token":"ACCESS_TOKEN",
"expires_in":7200,
"refresh_token":"REFRESH_TOKEN",
"openid":"OPENID",
"scope":"SCOPE",
"unionid": "o6_bmasdasdsad6_2sgVt7hMZOPfL" //这里一般没有返回,待定。
} //错误的时候:
{"errcode":40029,"errmsg":"invalid code"}
json解析的对象代码
public class WeiXinAccessTokenResult
{
public WeiXinAccessTokenModel SuccessResult { get; set; }
public bool Result { get; set; } public WeiXinErrorMsg ErrorResult { get; set; }
} /// <summary>
/// 通过code获取access_token 请求成功的实体
/// </summary>
public class WeiXinAccessTokenModel
{
/// <summary>
/// 接口调用凭证
/// </summary>
public string access_token { get; set; }
/// <summary>
/// access_token接口调用凭证超时时间,单位(秒)
/// </summary>
public int expires_in { get; set; }
/// <summary>
/// 用户刷新access_token
/// </summary>
public string refresh_token { get; set; }
/// <summary>
/// 授权用户唯一标识
/// </summary>
public string openid { get; set; }
/// <summary>
/// 用户授权的作用域,使用逗号(,)分隔
/// </summary>
public string scope { get; set; }
} /// <summary>
/// 微信错误访问的情况
/// </summary>
public class WeiXinErrorMsg
{
/// <summary>
/// 错误编号
/// </summary>
public int errcode { get; set; }
/// <summary>
/// 错误提示消息
/// </summary>
public string errmsg { get; set; }
}
Tool.HttpGet方法
public static string HttpGet(string Url, string postDataStr)
{
HttpWebRequest request = (HttpWebRequest)WebRequest.Create(Url + (postDataStr == "" ? "" : "?") + postDataStr);
request.Method = "GET";
request.ContentType = "text/html;charset=UTF-8"; HttpWebResponse response = (HttpWebResponse)request.GetResponse();
Stream myResponseStream = response.GetResponseStream();
StreamReader myStreamReader = new StreamReader(myResponseStream, Encoding.GetEncoding("utf-8"));
string retString = myStreamReader.ReadToEnd();
myStreamReader.Close();
myResponseStream.Close(); return retString;
}
3. 获取用户信息
获取用户很简单,只需要传一个token以及openid获取用户的基本信息。
代码如下:
string url = "https://api.weixin.qq.com/sns/userinfo";
string urlParam = "access_token=" + accessToken + "&openid=" + openId +
"⟨=zh_CN";
string jsonStr = Toos.HttpGet(url, urlParam); ;
WeiXinUserInfoResult result = new WeiXinUserInfoResult();
result.token = accessToken;
if (jsonStr.Contains("errcode"))
{
WeiXinErrorMsg errorResult = new WeiXinErrorMsg();
errorResult = JsonHelper.ParseFormJson<WeiXinErrorMsg>(jsonStr);
result.ErrorMsg = errorResult;
result.Result = false;
}
else
{
WXUserInfo userInfo = new WXUserInfo();
userInfo = JsonHelper.ParseFormJson<WXUserInfo>(jsonStr);
result.UserInfo = userInfo;
result.Result = true;
}
返回的json对象
{
"openid":" OPENID",
" nickname": NICKNAME,
"sex":"1",
"province":"PROVINCE"
"city":"CITY",
"country":"COUNTRY",
"headimgurl": "http://wx.qlogo.cn/mmopen/g3MonUZtNHkdmzicIlibx6iaFqAc56vxLSUfpb6n5WKSYVY0ChQKkiaJSgQ1dZuTOgvLLrhJbERQQ4eMsv84eavHiaiceqxibJxCfHe/46",
"privilege":[
"PRIVILEGE1"
"PRIVILEGE2"
],
"unionid": "o6_bmasdasdsad6_2sgVt7hMZOPfL"
}
解析json对象
public class WeiXinUserInfoResult
{ public string token { get; set; }
/// <summary>
/// 微信用户信息
/// </summary>
public WXUserInfo UserInfo { get; set; }
/// <summary>
/// 结果
/// </summary>
public bool Result { get; set; }
/// <summary>
/// 错误信息
/// </summary>
public WeiXinErrorMsg ErrorMsg { get; set; } public override string ToString()
{
return string.Format("UserInfo: {0}, Result: {1}, ErrorMsg: {2}, token:{3}", UserInfo, Result, ErrorMsg, token);
}
}
备注:
添加jsonhelp类
public class JsonHelper
{
public JsonHelper()
{
//
// TODO: Add constructor logic here
//
} /// <summary>
/// 把对象序列化 JSON 字符串
/// </summary>
/// <typeparam name="T">对象类型</typeparam>
/// <param name="obj">对象实体</param>
/// <returns>JSON字符串</returns>
public static string GetJson<T>(T obj)
{
//记住 添加引用 System.ServiceModel.Web
/**
* 如果不添加上面的引用,System.Runtime.Serialization.Json; Json是出不来的哦
* */
DataContractJsonSerializer json = new DataContractJsonSerializer(typeof (T));
using (MemoryStream ms = new MemoryStream())
{
json.WriteObject(ms, obj);
string szJson = Encoding.UTF8.GetString(ms.ToArray());
return szJson;
}
} /// <summary>
/// 把JSON字符串还原为对象
/// </summary>
/// <typeparam name="T">对象类型</typeparam>
/// <param name="szJson">JSON字符串</param>
/// <returns>对象实体</returns>
public static T ParseFormJson<T>(string szJson)
{
T obj = Activator.CreateInstance<T>();
using (MemoryStream ms = new MemoryStream(Encoding.UTF8.GetBytes(szJson)))
{
DataContractJsonSerializer dcj = new DataContractJsonSerializer(typeof (T));
return (T) dcj.ReadObject(ms);
}
}
}
详细请看微信官方说明:
http://qydev.weixin.qq.com/wiki/index.php?title=WeixinJS%E6%8E%A5%E5%8F%A3
可以关注本人的公众号,多年经验的原创文章共享给大家。

微信授权步骤与详解 -- c#篇的更多相关文章
- Redis详解入门篇
Redis详解入门篇 [本教程目录] 1.redis是什么2.redis的作者3.谁在使用redis4.学会安装redis5.学会启动redis6.使用redis客户端7.redis数据结构 – 简介 ...
- bt协议详解 基础篇(上)
bt协议详解 基础篇(上) 最近开发了一个免费教程的网站,产生了仔细了解bt协议的想法,所以写了这一篇文章,后续还会写一些关于搜索和索引的东西,都是在开发这个网站的过程中学习到的技术,敬请期待. 1 ...
- java实现微信扫一扫详解
java实现微信扫一扫详解 一.微信JS-SDK参数配置及查找 JS安全域名配置(查找:微信公众号里-公众号设置-功能设置页) 注:1.安全域名外网必须可以访问的到 2.域名不能有下划线 3.要将 ...
- Redis详解入门篇(转载)
Redis详解入门篇(转载) [本教程目录] 1.redis是什么2.redis的作者3.谁在使用redis4.学会安装redis5.学会启动redis6.使用redis客户端7.redis数据结构 ...
- RocketMQ源码详解 | Producer篇 · 其二:消息组成、发送链路
概述 在上一节 RocketMQ源码详解 | Producer篇 · 其一:Start,然后 Send 一条消息 中,我们了解了 Producer 在发送消息的流程.这次我们再来具体下看消息的构成与其 ...
- RocketMQ源码详解 | Broker篇 · 其四:事务消息、批量消息、延迟消息
概述 在上文中,我们讨论了消费者对于消息拉取的实现,对于 RocketMQ 这个黑盒的心脏部分,我们顺着消息的发送流程已经将其剖析了大半部分.本章我们不妨乘胜追击,接着讨论各种不同的消息的原理与实现. ...
- bt协议详解 DHT篇(下)
bt协议详解 DHT篇(下) 最近开发了一个免费教程的网站,产生了仔细了解bt协议的想法,这篇文章是bt协议详解系列的第三篇,后续还会写一些关于搜索和索引的东西,都是在开发这个网站的过程中学习到的技术 ...
- bt协议详解 DHT篇(上)
bt协议详解 DHT篇(上) 最近开发了一个免费教程的网站,突然产生了仔细了解bt协议的想法,这篇文章是bt协议详解系列的第三篇,后续还会写一些关于搜索和索引的东西,都是在开发这个网站的过程中学习到的 ...
- IIS负载均衡-Application Request Route详解第二篇:创建与配置Server Farm(转载)
IIS负载均衡-Application Request Route详解第二篇:创建与配置Server Farm 自从本系列发布之后,收到了很多的朋友的回复!非常感谢,同时很多朋友问到了一些问题,有些问 ...
随机推荐
- 如果因特网中的所有链路都提供可靠的交付服务,TCP可靠传输服务是多余的吗?
IP协议因为是无连接的, 所以其传输是不可靠的.虽然链路保证了数据包在端到端的传输中不发生差错,但是它不能保证IP数据包是按照正确的书需到达最终的目的地.IP数据包可以使用不同的路由通过网络,到达接收 ...
- Java String字符串方法
1.String构造函数 1> String() 2> String(char[] chars) String(char[] chars,int startIndex,int numCha ...
- mvc之文件下载
首先你要有四张图片,也就是数组中的数 public ActionResult Index()//创建视图{ViewBag.list =new int[] { 5, 6, 7,8 };return Vi ...
- 阿里云安装Tomcat
1.Apache官方网站下载Tomcat http://mirrors.hust.edu.cn/apache/tomcat/tomcat-8/v8.0.35/bin/apache-tomcat-8.0 ...
- Xenu Link Sleuth-简单使用
1.工具说明 xenu link sleuth,主要用于测试网站死链接.包括图片.链接. 下载地址:http://home.snafu.de/tilman/xenulink.html#Download ...
- Ubuntu下移植OpenCv
通过近一周的时候终于成功交叉编译opencv成功了,真心不容易.有一句话乃真理也,凡事贵在坚持.过程总是痛苦的,因为不懂得很多问题但是又需要面对很多问题,最大的收获就是耐心解决所有问题后就懂得这些了. ...
- 8.4.2 Fresco
Fresco是Facebook公司的黑科技:http://fresco-cn.org/ 真三级缓存,变换后的BItmap(内存),变换前的原始图片(内存),硬盘缓存.在内存管理上做到了极致.对于重度图 ...
- java-注解
概念 Annontation是Java5开始引入的新特征.中文名称一般叫注解.它提供了一种安全的类似注释的机制,用来将任何的信息或元数据与程序元素(类.方法.成员变量等)进行关联.更通俗的意思是为程序 ...
- 工作随笔——UIButton的EdgeInsets + Swift中的正则表达式;
1.UIButton的EdgeInsets UIButton的EdgeInsets方法,是用来设置title和image对于上左下右四个方向的偏移,但是很奇怪的是,刚开始只有Image,titile也 ...
- js禁止重复提交方法
beforeSend: function () { // 禁用按钮防止重复提交 $("#fileForm a[class='btn blue_btn']").removeAttr( ...