您好,登錄后才能下訂單哦!
如何在ThinkPHP項(xiàng)目中開發(fā)一個(gè)支付寶接口功能?很多新手對此不是很清楚,為了幫助大家解決這個(gè)難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。
一、下載支付寶接口包
下載地址:https://b.alipay.com/order/productDetail.htm?productId=2012111200373124&tabId=4#ps-tabinfo-hash
具體如何下載,我就不在羅嗦了~~
二、重新整理接口包文件,這一步應(yīng)該算是比較關(guān)鍵的(個(gè)人認(rèn)為),下載下來的接口包文件有很多語言的源碼
我們選擇 create_direct_pay_by_user-PHP-UTF-8 這個(gè)名稱的接口文件,里面包括如下文件:
images文件里是支付寶相關(guān)的一些標(biāo)志的圖片,我們暫不管他,lib文件很重要,是整個(gè)接口的核心類文件;
alipay.config.php是相關(guān)參數(shù)的配置文件
alipayapi.php 是支付寶接口入口文件
notify_url.php 是服務(wù)器異步通知頁面文件;
return_url.php 是頁面跳轉(zhuǎn)同步通知文件;
在ThinkPHP的框架文件下,找到Extend 進(jìn)入,再進(jìn)入Vendor,在Vendor文件夾下,新建文件夾Alipay,把支付寶作為第三方類庫引入,然后,復(fù)制支付寶接口文件包中l(wèi)ib文件里的所有文件,一共4個(gè)文件,如下:
現(xiàn)在對以上文件進(jìn)行重命名.
alipay_core.function.php重命名為:Corefunction.php;
alipay_md5.function.php重命名為:Md5function.php;
alipay_notify.class.php重命名為:Notify.php;
alipay_submit.class.php重命名為:Submit.php;
然后,打開Submit.php文件,把以下代碼去掉;
require_once("alipay_core.function.php");
require_once("alipay_md5.function.php");同樣,打開Notify.php文件,把以下兩段代碼去掉require_once("alipay_core.function.php");
require_once("alipay_md5.function.php");為什么要去掉以上兩個(gè)文件中的這兩段代碼,因?yàn)樵陧?xiàng)目中調(diào)用接口文件的時(shí)候,我把所有4個(gè)核心文件都通過vendor來進(jìn)行引入。所以,這不再需要導(dǎo)入。
到此,支付寶接口包相關(guān)核心類庫的整理基本完成?,F(xiàn)在開始在項(xiàng)目中調(diào)用;
三、在項(xiàng)目中調(diào)用支付寶接口
調(diào)用分兩步:
1、在配置文件中Conf/Config.php文件中對支付寶相關(guān)參數(shù)進(jìn)行配置
復(fù)制代碼 代碼如下:
//支付寶配置參數(shù)
'alipay_config'=>array(
'partner' =>'20********50', //這里是你在成功申請支付寶接口后獲取到的PID;
'key'=>'9t***********ie',//這里是你在成功申請支付寶接口后獲取到的Key
'sign_type'=>strtoupper('MD5'),
'input_charset'=> strtolower('utf-8'),
'cacert'=> getcwd().'\\cacert.pem',
'transport'=> 'http',
),
//以上配置項(xiàng),是從接口包中alipay.config.php 文件中復(fù)制過來,進(jìn)行配置;
'alipay' =>array(
//這里是賣家的支付寶賬號,也就是你申請接口時(shí)注冊的支付寶賬號
'seller_email'=>'pay@xxx.com',
//這里是異步通知頁面url,提交到項(xiàng)目的Pay控制器的notifyurl方法;
'notify_url'=>'http://www.xxx.com/Pay/notifyurl',
//這里是頁面跳轉(zhuǎn)通知url,提交到項(xiàng)目的Pay控制器的returnurl方法;
'return_url'=>'http://www.xxx.com/Pay/returnurl',
//支付成功跳轉(zhuǎn)到的頁面,我這里跳轉(zhuǎn)到項(xiàng)目的User控制器,myorder方法,并傳參payed(已支付列表)
'successpage'=>'User/myorder?ordtype=payed',
//支付失敗跳轉(zhuǎn)到的頁面,我這里跳轉(zhuǎn)到項(xiàng)目的User控制器,myorder方法,并傳參unpay(未支付列表)
'errorpage'=>'User/myorder?ordtype=unpay',
),
2、新建一個(gè)PayAction控制器代碼如下
復(fù)制代碼 代碼如下:
<?php
class PayAction extends Action{
//在類初始化方法中,引入相關(guān)類庫
public function _initialize() {
vendor('Alipay.Corefunction');
vendor('Alipay.Md5function');
vendor('Alipay.Notify');
vendor('Alipay.Submit');
}
//doalipay方法
public function doalipay(){
// require_once("alipay.config.php");
// require_once("lib/alipay_submit.class.php");
//這里我們通過TP的C函數(shù)把配置項(xiàng)參數(shù)讀出,賦給$alipay_config;
$alipay_config=C('alipay_config');
$payment_type = "1"; //支付類型 //必填,不能修改
$notify_url = C('alipay.notify_url'); //服務(wù)器異步通知頁面路徑
$return_url = C('alipay.return_url'); //頁面跳轉(zhuǎn)同步通知頁面路徑
$seller_email = C('alipay.seller_email');//賣家支付寶帳戶必填
$out_trade_no = $_POST['trade_no'];//商戶訂單號 通過支付頁面的表單進(jìn)行傳遞,注意要唯一!
$subject = $_POST['ordsubject']; //訂單名稱 //必填 通過支付頁面的表單進(jìn)行傳遞
$total_fee = $_POST['ordtotal_fee']; //付款金額 //必填 通過支付頁面的表單進(jìn)行傳遞
$body = $_POST['ordbody']; //訂單描述 通過支付頁面的表單進(jìn)行傳遞
$show_url = $_POST['ordshow_url']; //商品展示地址 通過支付頁面的表單進(jìn)行傳遞
$anti_phishing_key = "";//防釣魚時(shí)間戳 //若要使用請調(diào)用類文件submit中的query_timestamp函數(shù)
$exter_invoke_ip = get_client_ip(); //客戶端的IP地址
//構(gòu)造要請求的參數(shù)數(shù)組,無需改動
$parameter = array(
"service" => "create_direct_pay_by_user",
"partner" => trim($alipay_config['partner']),
"payment_type" => $payment_type,
"notify_url" => $notify_url,
"return_url" => $return_url,
"seller_email" => $seller_email,
"out_trade_no" => $out_trade_no,
"subject" => $subject,
"total_fee" => $total_fee,
"body" => $body,
"show_url" => $show_url,
"anti_phishing_key" => $anti_phishing_key,
"exter_invoke_ip" => $exter_invoke_ip,
"_input_charset" => trim(strtolower($alipay_config['input_charset']))
);
//建立請求
$alipaySubmit = new AlipaySubmit($alipay_config);
$html_text = $alipaySubmit->buildRequestForm($parameter,"post", "確認(rèn)");
echo $html_text;
}
function notifyurl(){
//require_once("alipay.config.php");
//require_once("lib/alipay_notify.class.php");
//這里還是通過C函數(shù)來讀取配置項(xiàng),賦值給$alipay_config
$alipay_config=C('alipay_config');
//計(jì)算得出通知驗(yàn)證結(jié)果
$alipayNotify = new AlipayNotify($alipay_config);
$verify_result = $alipayNotify->verifyNotify();
if($verify_result) {
//驗(yàn)證成功
//獲取支付寶的通知返回參數(shù),可參考技術(shù)文檔中服務(wù)器異步通知參數(shù)列表
$out_trade_no = $_POST['out_trade_no']; //商戶訂單號
$trade_no = $_POST['trade_no']; //支付寶交易號
$trade_status = $_POST['trade_status']; //交易狀態(tài)
$total_fee = $_POST['total_fee']; //交易金額
$notify_id = $_POST['notify_id']; //通知校驗(yàn)ID。
$notify_time = $_POST['notify_time']; //通知的發(fā)送時(shí)間。格式為yyyy-MM-dd HH:mm:ss。
$buyer_email = $_POST['buyer_email']; //買家支付寶帳號;
$parameter = array(
"out_trade_no" => $out_trade_no, //商戶訂單編號;
"trade_no" => $trade_no, //支付寶交易號;
"total_fee" => $total_fee, //交易金額;
"trade_status" => $trade_status, //交易狀態(tài)
"notify_id" => $notify_id, //通知校驗(yàn)ID。
"notify_time" => $notify_time, //通知的發(fā)送時(shí)間。
"buyer_email" => $buyer_email, //買家支付寶帳號;
);
if($_POST['trade_status'] == 'TRADE_FINISHED') {
//
}else if ($_POST['trade_status'] == 'TRADE_SUCCESS') { if(!checkorderstatus($out_trade_no)){
orderhandle($parameter);
//進(jìn)行訂單處理,并傳送從支付寶返回的參數(shù);
}
}
echo "success"; //請不要修改或刪除
}else {
//驗(yàn)證失敗
echo "fail";
}
}
function returnurl(){
//頭部的處理跟上面兩個(gè)方法一樣,這里不羅嗦了!
$alipay_config=C('alipay_config');
$alipayNotify = new AlipayNotify($alipay_config);//計(jì)算得出通知驗(yàn)證結(jié)果
$verify_result = $alipayNotify->verifyReturn();
if($verify_result) {
//驗(yàn)證成功
//獲取支付寶的通知返回參數(shù),可參考技術(shù)文檔中頁面跳轉(zhuǎn)同步通知參數(shù)列表
$out_trade_no = $_GET['out_trade_no']; //商戶訂單號
$trade_no = $_GET['trade_no']; //支付寶交易號
$trade_status = $_GET['trade_status']; //交易狀態(tài)
$total_fee = $_GET['total_fee']; //交易金額
$notify_id = $_GET['notify_id']; //通知校驗(yàn)ID。
$notify_time = $_GET['notify_time']; //通知的發(fā)送時(shí)間。
$buyer_email = $_GET['buyer_email']; //買家支付寶帳號;
$parameter = array(
"out_trade_no" => $out_trade_no, //商戶訂單編號;
"trade_no" => $trade_no, //支付寶交易號;
"total_fee" => $total_fee, //交易金額;
"trade_status" => $trade_status, //交易狀態(tài)
"notify_id" => $notify_id, //通知校驗(yàn)ID。
"notify_time" => $notify_time, //通知的發(fā)送時(shí)間。
"buyer_email" => $buyer_email, //買家支付寶帳號
);
if($_GET['trade_status'] == 'TRADE_FINISHED' || $_GET['trade_status'] == 'TRADE_SUCCESS') {
if(!checkorderstatus($out_trade_no)){
orderhandle($parameter); //進(jìn)行訂單處理,并傳送從支付寶返回的參數(shù);
}
$this->redirect(C('alipay.successpage'));//跳轉(zhuǎn)到配置項(xiàng)中配置的支付成功頁面;
}else {
echo "trade_status=".$_GET['trade_status'];
$this->redirect(C('alipay.errorpage'));//跳轉(zhuǎn)到配置項(xiàng)中配置的支付失敗頁面;
}
}else {
//驗(yàn)證失敗
//如要調(diào)試,請看alipay_notify.php頁面的verifyReturn函數(shù)
echo "支付失??!";
}
}
}
?>
3、這里有幾個(gè)支付處理過程中需要用到的函數(shù),我把這些函數(shù)寫到了項(xiàng)目的Common/common.php中,這樣不用手動調(diào)用,即可直接使用這些函數(shù),代碼如下:
復(fù)制代碼 代碼如下:
//Orderlist數(shù)據(jù)表,用于保存用戶的購買訂單記錄;
//在線交易訂單支付處理函數(shù)
//函數(shù)功能:根據(jù)支付接口傳回的數(shù)據(jù)判斷該訂單是否已經(jīng)支付成功;
//返回值:如果訂單已經(jīng)成功支付,返回true,否則返回false;
function checkorderstatus($ordid){
$Ord=M('Orderlist');
$ordstatus=$Ord->where('ordid='.$ordid)->getField('ordstatus');
if($ordstatus==1){
return true;
}else{
return false;
}
}
//處理訂單函數(shù)
//更新訂單狀態(tài),寫入訂單支付后返回的數(shù)據(jù)
function orderhandle($parameter){
$ordid=$parameter['out_trade_no'];
$data['payment_trade_no'] =$parameter['trade_no'];
$data['payment_trade_status'] =$parameter['trade_status'];
$data['payment_notify_id'] =$parameter['notify_id'];
$data['payment_notify_time'] =$parameter['notify_time'];
$data['payment_buyer_email'] =$parameter['buyer_email'];
$data['ordstatus'] =1;
$Ord=M('Orderlist');
$Ord->where('ordid='.$ordid)->save($data);
}
//獲取一個(gè)隨機(jī)且唯一的訂單號;
function getordcode(){
$Ord=M('Orderlist');
$numbers = range (10,99);
shuffle ($numbers);
$code=array_slice($numbers,0,4);
$ordcode=$code[0].$code[1].$code[2].$code[3];
$oldcode=$Ord->where("ordcode='".$ordcode."'")->getField('ordcode');
if($oldcode){
getordcode();
}else{
return $ordcode;
}
}
四、總結(jié)幾點(diǎn)
1、接口包中l(wèi)ib文件中的文件復(fù)制到Vendor后,重命名為TP規(guī)范的命名規(guī)則,為的是調(diào)用方便,當(dāng)然你要改成其他名稱也可以;
2、把執(zhí)行支付操作(doalipay),處理異步返回結(jié)果(notifyurl),處理跳轉(zhuǎn)返回結(jié)果(returnurl)三個(gè)支付接口的核心頁面寫到一個(gè)PayAction控制器中。
3、提交支付的頁面中,可以在提交之前先把一些參數(shù)要傳遞的內(nèi)容先通過隱藏域的方法組合好,比如金額先計(jì)算好,訂單名稱,訂單描述等先用字符串組合好。然后提交表單,這樣,在doalipay方法中只要直接構(gòu)造傳遞參數(shù),直接進(jìn)行提交就行過了。
4、支付返回后的處理因?yàn)橐诋惒胶吞D(zhuǎn)兩個(gè)方法中都要進(jìn)行相應(yīng)的判斷和處理,所以,把這些判斷和處理寫到一個(gè)自定義函數(shù)中,這樣只要調(diào)用函數(shù)即可,使得代碼更加清晰明了。
5、notify_url和return_url兩種模式的返回url必須使用http://xxxxxxx這樣的絕對路徑,因?yàn)槔锸菑闹Ц秾毱脚_返回到你的項(xiàng)目頁面,不能使用相對路徑。
看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進(jìn)一步的了解或閱讀更多相關(guān)文章,請關(guān)注億速云行業(yè)資訊頻道,感謝您對億速云的支持。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。