本文主要和大家分享PHP實(shí)現(xiàn)微信支付流程锅知,最近接觸到一個(gè)項(xiàng)目,涉及到微信支付脓钾,搞微信開發(fā)這么久以來售睹,還沒搞過支付,之前也就搞過公眾號(hào)發(fā)紅包可训,感謝前輩們的探索昌妹,我看了他們的博文,讓我少走了很多彎路握截。
前期準(zhǔn)備:
1.微信認(rèn)證服務(wù)號(hào)飞崖,并且開通了微信支付
2.微信支付SDK,下載地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=11_1
3.登錄微信支付平臺(tái)https://pay.weixin.qq.com/index.php/account/api_cert下載支付證書
方法步驟:
1.demo文件處理
(1)將官方的demo下載下來谨胞,文件名為WxpayAPI_php_v3固歪,把這文件重命名為wxpay,為了后邊書寫目錄方便胯努;
(2)打開lib文件夾下的WxPay.Api.php文件牢裳,在537行有一段curl網(wǎng)絡(luò)請(qǐng)求配置代碼:
curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,TRUE);
curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,2);//嚴(yán)格校驗(yàn)
替換成:
curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,2);//嚴(yán)格校驗(yàn)
為了禁止 cURL 驗(yàn)證對(duì)等證書(peer's certificate)。
(3)打開lib文件夾下的WxPay.Config.php文件叶沛,第25行開始蒲讯,根據(jù)自己的賬號(hào)完成基本信息設(shè)置;
const?APPID?=?'公眾賬號(hào)APPID';
const?MCHID?=?'商戶號(hào)';
const?KEY?=?'商戶支付密鑰';
const?APPSECRET?=?'公眾帳號(hào)secert';
(4)打開lib文件夾下的WxPay.Notify.php文件恬汁,第79行的代碼:
if($needSign?==?true?&&
$this->GetReturn_code($return_code)?==?"SUCCESS")
{
$this->SetSign();
}
改成:
if($needSign?==?true?&&
$this->GetReturn_code()?==?"SUCCESS")
{
$this->SetSign();
}
(5)打開cert證書目錄伶椿,將里邊的兩個(gè)證書換成自己的支付證書。
2.公眾號(hào)后臺(tái)設(shè)置
(1)配置網(wǎng)頁(yè)授權(quán)域名氓侧,我的域名是(xy.chuyin.ren)脊另;
(1)配置支付授權(quán)目錄,域名是(xy.chuyin.ren)约巷,我將demo放到此域名指向的目錄的weixinopen/文件夾下偎痛,demo中jsapi.php文件位于example/目錄下,所以支付授權(quán)目錄為:xy.chuyin.ren/weixinopen/wxpay/example/
3.支付流程
打開example目錄下的jsapi.php文件独郎,支付發(fā)起和處理踩麦,都是在這里完成。
(1)獲取用戶openid
之前配置好了自己的APPID和APPSecert氓癌,所以這里不用處理谓谦。
//①、獲取用戶openid
$tools?=?new?JsApiPay();
$openId?=?$tools->GetOpenid();
這里首先初始化的一個(gè)JsApiPay()類得到一個(gè)對(duì)象贪婉,文件對(duì)應(yīng)example/目錄下的WxPay.JsApiPay.php反粥,調(diào)用GetOpenid()方法,會(huì)自動(dòng)獲取自己的openID疲迂。
(2)統(tǒng)一下單
//②才顿、統(tǒng)一下單
$input?=?new?WxPayUnifiedOrder();
$input->SetBody("test");
$input->SetAttach("test");
$input->SetOut_trade_no(WxPayConfig::MCHID.date("YmdHis"));
$input->SetTotal_fee("1");
$input->SetTime_start(date("YmdHis"));
$input->SetTime_expire(date("YmdHis",?time()?+?600));
$input->SetGoods_tag("test");
$input->SetNotify_url("http://paysdk.weixin.qq.com/example/notify.php");
$input->SetTrade_type("JSAPI");
$input->SetOpenid($openId);
$order?=?WxPayApi::unifiedOrder($input);
echo?'統(tǒng)一下單支付單信息
';
printf_info($order);
$jsApiParameters?=?$tools->GetJsApiParameters($order);
對(duì)應(yīng)WxPay.Api.php的第24行的unifiedOrder()方法,配置訂單信息和支付回調(diào)函數(shù)尤蒿,這里需要修改幾個(gè)參數(shù):
A. 商品名稱:
$input->SetBody("test");
B. 訂單號(hào)
$input->SetOut_trade_no(WxPayConfig::MCHID.date("YmdHis"));
C. 支付金額
$input->SetTotal_fee("1");
D. 支付驗(yàn)證鏈接
設(shè)置為你的notify.php文件所在的位置郑气,所以我這里設(shè)置為:http://xy.chuyin.ren/weixinopen/wxpay/example/notify.php
也可以寫其他地址,當(dāng)然要在支付授權(quán)域名之下,支付成功之后就會(huì)自動(dòng)回調(diào)到該鏈接指定的方法里邊,可以在里邊進(jìn)行判斷和數(shù)據(jù)庫(kù)操作.
$input->SetNotify_url("http://paysdk.weixin.qq.com/example/notify.php");
E. 附加參數(shù)
$input->SetAttach("test");
附加參數(shù),可填可不填,填寫的話,里邊字符串最好不要出現(xiàn)空格。
這時(shí)候腰池,點(diǎn)擊支付應(yīng)該就可以成功支付了尾组。
(3)發(fā)起支付
//調(diào)用微信JS?api?支付
function?jsApiCall()
{
WeixinJSBridge.invoke(
'getBrandWCPayRequest',
,
function(res){
WeixinJSBridge.log(res.err_msg);
alert(res.err_code+res.err_desc+res.err_msg);
}
);
}
function?callpay()
{
if?(typeof?WeixinJSBridge?==?"undefined"){
if(?document.addEventListener?){
document.addEventListener('WeixinJSBridgeReady',?jsApiCall,?false);
}else?if?(document.attachEvent){
document.attachEvent('WeixinJSBridgeReady',?jsApiCall);
document.attachEvent('onWeixinJSBridgeReady',?jsApiCall);
}
}else{
jsApiCall();
}
}
點(diǎn)擊立即支付按鈕調(diào)用的就是 callpay() 函數(shù),他有會(huì)調(diào)用jsApiCall() 函數(shù)打開支付程序巩螃。
jsApiCall() 函數(shù)會(huì)監(jiān)聽每一步動(dòng)作:
res.err_msg 為get_brand_wcpay_request:cancel 表明前端判斷的取消支付演怎,es.err_msg 為get_brand_wcpay_request:ok 表明前端判斷的支付成功,我們可以根據(jù)這個(gè)將支付跳轉(zhuǎn)到成功頁(yè)面避乏。
(4)支持成功回調(diào)
通過前端jsApiCall()函數(shù)可以監(jiān)聽支付結(jié)果爷耀,但是這個(gè)并不可信。確認(rèn)是否支付成功還是應(yīng)當(dāng)通過notify.php 處理業(yè)務(wù)邏輯拍皮。前邊配置好了支付驗(yàn)證鏈接SetNotify_url()歹叮,支付完成后,微信服務(wù)器會(huì)根據(jù)鏈接自動(dòng)請(qǐng)求你的notify.php文件,打開這個(gè)文件铆帽,其實(shí)這個(gè)文件最主要的代碼就兩行:
$notify?=?new?PayNotifyCallBack();
$notify->Handle(false);
/**
*
*?回調(diào)入口
*?@param?bool?$needSign??是否需要簽名輸出
*/
final?public?function?Handle($needSign?=?true)
{
$msg?=?"OK";
//當(dāng)返回false的時(shí)候咆耿,表示notify中調(diào)用NotifyCallBack回調(diào)失敗獲取簽名校驗(yàn)失敗,此時(shí)直接回復(fù)失敗
$result?=?WxpayApi::notify(array($this,?'NotifyCallBack'),?$msg);
if($result?==?false){
$this->SetReturn_code("FAIL");
$this->SetReturn_msg($msg);
$this->ReplyNotify(false);
return;
}?else?{
//該分支在成功回調(diào)到NotifyCallBack方法爹橱,處理完成之后流程
$this->SetReturn_code("SUCCESS");
$this->SetReturn_msg("OK");
}
$this->ReplyNotify($needSign);
}
由此跟蹤到WxPay.Notify.php類文件的Handle()函數(shù):
主要代碼:
$result?=?WxpayApi::notify(array($this,?'NotifyCallBack'),?$msg);
然后來到WxPay.Api.php文件的第411行萨螺,notify()函數(shù):
/**
*
*?支付結(jié)果通用通知
*?@param?function?$callback
*?直接回調(diào)函數(shù)使用方法:?notify(you_function);
*?回調(diào)類成員函數(shù)方法:notify(array($this,?you_function));
*?$callback??原型為:function?function_name($data){}
*/
public?static?function?notify($callback,?&$msg)
{
//獲取通知的數(shù)據(jù)
$xml?=?$GLOBALS['HTTP_RAW_POST_DATA'];
//file_put_contents('log.txt',$xml,FILE_APPEND);
//如果返回成功則驗(yàn)證簽名
try?{
$result?=?WxPayResults::Init($xml);
}?catch?(WxPayException?$e){
$msg?=?$e->errorMessage();
return?false;
}
return?call_user_func($callback,?$result);
}
這里面的$xml=$GLOBALS['HTTP_RAW_POST_DATA'],就是支付成功后用戶返回給你的一個(gè)結(jié)果,他是一個(gè)xml格式的字符串慰技。
我們可以將這里返回的xml數(shù)據(jù)記錄下來椭盏,然后打開看看$out_trade_no就是在支付之前我自己設(shè)置的訂單號(hào)碼,$attach就是設(shè)置的附加參數(shù)吻商。
得到了這個(gè)訂單號(hào)掏颊,然后我就直接在下面寫支付成功后的邏輯了,比如改變數(shù)據(jù)庫(kù)中的數(shù)據(jù)等等艾帐。
這樣 微信支付的 JsApi支付就大致分析完成了乌叶。
這是集成了官方的SDK實(shí)現(xiàn)的,如果不使用SDK,可以使用更簡(jiǎn)單的方法,見:PHP實(shí)現(xiàn)微信支付(jsapi支付)和退款(無需集成支付SDK)