php 防止access token过期,微信調用接口,防止Access_token過期的方法

本文探讨了微信公众号调用用户接口时频繁获取Access_token的问题,提出通过文本文件或数据库缓存来减少请求次数,降低访问限制。作者分享了如何封装方法并利用SAE Storage持久化AccessToken,以解决调用频率限制和延迟问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

大家都知道,微信中調用訂閱用戶接口中需要Access_token,而根據微信官方文檔中說明:

access_token是公眾號的全局唯一票據,公眾號調用各接口時都需使用access_token。正常情況下access_token有效期為7200秒,重復獲取將導致上次獲取的access_token失效

所以這就導致我們每次去獲取訂閱用戶相關信息的時候都去重新取access_token。這就造成兩個問題。

1. 慢,因為取access_token是遠程在騰訊提供的API網址上,所以會有一定的延遲。

2. 訪問次數越多,超出接口調用限制,會被騰訊限制。我就遭過。后來查API手冊才知道有這句話:

默認每個公眾帳號都不能超過下面的頻率限制。 當超出調用接口頻率限制,調用對應接口將會收到如下錯誤信息:

{"errcode":45009,"errmsg":"api freq out of limit"}

接口名稱 頻率限制

獲取憑證接口 200(次/天)

自定義菜單創建接口 100(次/天)

自定義菜單查詢接口 1000(次/天)

自定義菜單刪除接口 100(次/天)

所以針對此問題,我想出了用文本文件或XML來存取動態的access_token。反正2小時才過期,寫入又不頻繁。當然你也可以存取到數據庫。

同樣的我用到了SAE的Storage

375ba6b97128e694787870457f56d90b.jpe

然后對訪問用戶相關信息和得到AccessToken的類 進行了封裝:

//得到訂閱用戶 (返回數組)

public function GetUserList()

{

$strjson = $this -> GetUrlReturn("https://api.weixin.qq.com/cgi-bin/user/get?access_token=%s");

$openidarr= $strjson->data->openid;

//print_r($openidarr); 調試

return $openidarr;

}

//得到訂閱用戶詳情(返回對象)

public function GetUserDetail($openid)

{

$url = "https://api.weixin.qq.com/cgi-bin/user/info?access_token=%s&openid={$openid}";

$strjson = $this -> GetUrlReturn($url);

return $strjson;

}

/*

*

* 私有成員變量 存token值

* 因為//access_token是公眾號的全局唯一票據,公眾號調用各接口時都需使用access_token。

* 正常情況下access_token有效期為7200秒,重復獲取將導致上次獲取的access_token失效。

*/

private $_token ;

/*

*

* 私有方法

*

*/

//得到Token對象並寫入到配置文件

private function InitToken()

{

$url = sprintf("https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s",APPID, SECRET);

//echo APPID;

$ch = curl_init(); //創建一個新url資源

curl_setopt($ch, CURLOPT_URL,$url);

curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);

$a = curl_exec($ch);

$strjson=json_decode($a);

$token = $strjson->access_token;

if (empty($token))

{

//修改 {"errcode":45009,"errmsg":"api freq out of limit"}

echo "錯誤:取得token無效,可能是調用太頻繁!"; //$strjson

throw new Exception('錯誤:取得token無效');

}

$obj = fopen("saestor://weixindata/token.txt","w+"); //SAE禁用fopen本地文件,這里需要Storage

fwrite($obj,$token);

$this -> _token = $token;

}

//封裝私有方法,調用得到Get的參數,$needToken默認為false, 不取值,這里有一個潛規則,%s為 self::$_token

private function GetUrlReturn($url, $needToken = false)

{

//第一次為空,則從文件中讀取

if (empty($this -> _token))

{

$obj = fopen("saestor://weixindata/token.txt","r");

$this -> _token = fgets($obj,1000);

}

//為空則重新取值

if (empty($this -> _token) || $needToken)

{

$this ->InitToken();

}

$newurl = sprintf($url, $this -> _token);

$ch = curl_init(); //創建一個新url資源

curl_setopt($ch, CURLOPT_URL,$newurl);

curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);

$a = curl_exec($ch);

$strjson=json_decode($a);

//var_dump($strjson); //開啟可調試

if (!empty($strjson-> errcode))

{

switch ($strjson-> errcode){

case 40001:

$this -> GetUrlReturn($url, true); //重新取值,可能是過期導致

break;

case 41001:

throw new Exception("缺少access_token參數:".$strjson->errmsg);

break;

default:

throw new Exception($strjson->errmsg); //其他錯誤,拋出

break;

}

}

return $strjson;

}大概意思是,默認就是從文本文件中取這個值,當這個值過期的時候,重新調用 InitToken 這個方法,獲得最新的AccessToken。這樣AccessToken就永遠不會過期了。

這是我想到的辦法,歡迎大家給出更好的方法,相互學習,相互提高。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值