一、什麼是Access_token?
access_token是公眾號的全局唯一票據,公眾號調用各接口時都需使用access_token。正常情況下access_token有效期為7200秒,重復獲取將導致上次獲取的access_token失效。由於獲取access_token的api調用次數非常有限,建議開發者全局存儲與更新access_token,頻繁刷新Access_token會導致api調用受限,影響自身業務。
二、要解決的問題
1、如何獲取Access_token。
2、由於access_token的有效期為7200秒,即2小時,並且重復獲取將導致上次獲取的access_token失效,獲取access_token的api調用次數非常有限,所以要解決如何全局存儲與更新Access_token。
三、思路
1、將Access_token存儲在數據庫中。
2、何時更新access_token呢?當access_token失效的時候更新,那麼怎麼判斷access_token有沒有失效呢?使用當前的access_token請求微信接口,獲取自定義菜單,如果返回的errcode為42001,則說明access_token已經失效,這時再重新獲取Access_token。
數據庫設計(表名SWX_Config):
四、代碼:
1、Http請求代碼(HttpRequestUtil類):
#region 請求Url,不發送數據
/// <summary>
/// 請求Url,不發送數據
/// </summary>
public
static
string
RequestUrl(
string
url)
{
return
RequestUrl(url,
"POST"
);
}
#endregion
#region 請求Url,不發送數據
/// <summary>
/// 請求Url,不發送數據
/// </summary>
public
static
string
RequestUrl(
string
url,
string
method)
{
// 設置參數
HttpWebRequest request = WebRequest.Create(url)
as
HttpWebRequest;
CookieContainer cookIEContainer =
new
CookIEContainer();
request.CookieContainer = cookIEContainer;
request.AllowAutoRedirect =
true
;
request.Method = method;
request.ContentType =
"text/Html"
;
request.Headers.Add(
"charset"
,
"utf-8"
);
//發送請求並獲取相應回應數據
HttpWebResponse response = request.GetResponse()
as
HttpWebResponse;
//直到request.GetResponse()程序才開始向目標網頁發送Post請求
Stream responseStream = response.GetResponseStream();
StreamReader sr =
new
StreamReader(responseStream, Encoding.UTF8);
//返回結果網頁(Html)代碼
string
content = sr.ReadToEnd();
return
content;
}
#endregion
2、輔助方法(Tools類):
namespace
SWX.Utils
{
/// <summary>
/// 工具類
/// </summary>
public
class
Tools
{
#region 獲取JSon字符串某節點的值
/// <summary>
/// 獲取JSon字符串某節點的值
/// </summary>
public
static
string
GetJSonValue(
string
JSonStr,
string
key)
{
string
result =
string
.Empty;
if
(!
string
.IsNullOrEmpty(JSonStr))
{
key =
"\""
+ key.Trim(
'"'
) + "\
""
;
int
index = JSonStr.IndexOf(key) + key.Length + 1;
if
(index > key.Length + 1)
{
//先截逗號,若是最後一個,截“}”號,取最小值
int
end = JSonStr.IndexOf(
','
, index);
if
(end == -1)
{
end = JSonStr.IndexOf(
'}'
, index);
}
result = JSonStr.Substring(index, end - index);
result = result.Trim(
new
char
[] {
'"'
,
' '
,
'\''
});
//過濾引號或空格
}
}
return
result;
}
#endregion
}
}
3、判斷Access_token是否過期(WXApi類):
#region 驗證Token是否過期
/// <summary>
/// 驗證Token是否過期
/// </summary>
public
static
bool
TokenExpired(
string
Access_token)
{
string
JSonStr = HttpRequestUtil.RequestUrl(
string
.Format(
"https://api.weixin.QQ.com/CGI-bin/menu/get?Access_token={0}"
, Access_token));
if
(Tools.GetJsonValue(JSonStr,
"errcode"
) ==
"42001"
)
{
return
true
;
}
return
false
;
}
#endregion
4、請求微信接口,獲取Access_token(WXApi類):
#region 獲取Token
/// <summary>
/// 獲取Token
/// </summary>
public
static
string
GetToken(
string
appid,
string
secret)
{
string
strJSon = HttpRequestUtil.RequestUrl(
string
.Format(
"https://api.weixin.QQ.com/CGI-bin/token?grant_type=clIEnt_credential&appid={0}&secret={1}"
, appid, secret));
return
Tools.GetJsonValue(strJSon,
"Access_token"
);
}
#endregion
5、全局存儲與更新Access_token(AdminUtil類):
#region 獲取Access_token
/// <summary>
/// 獲取Access_token
/// </summary>
public
static
string
GetAccessToken(PageBase page)
{
string
Access_token =
string
.Empty;
UserInfo user = GetLoginUser(page);
if
(user !=
null
)
{
if
(
string
.IsNullOrWhiteSpace(user.Access_token))
//尚未保存過Access_token
{
Access_token = WXApi.GetToken(user.AppID, user.APPSecret);
}
else
{
if
(WXApi.TokenExpired(user.Access_token))
//Access_token過期
{
Access_token = WXApi.GetToken(user.AppID, user.APPSecret);
}
else
{
return
user.Access_token;
}
}
MSSQLHelper.ExecuteSql(
string
.Format(
"update SWX_Config set Access_token='{0}' where UserName='{1}'"
, Access_token, user.UserName));
}
return
Access_token;
}
#endregion