ThinkPHP接入支付宝支付功能

最近做系统,需要实现在线支付功能,毫不犹豫,选择的是支付宝的接口支付功能。这里我用的是即时到帐的接口,具体实现的步骤如下:

一、下载支付宝接口包

下载地址:

https://b.alipay.com/order/productDetail.htm?productId=2012111200373124&tabId=4#ps-tabinfo-hash

具体如何下载,我就不在罗嗦了~~

很多人反映,用支付宝的接口到最后面会出现验证错误。其实,这里需要对接口程序进行一下改造。需要添加几个自定义函数。为了让大家以后避免出现同样的问题,我把我改造好的支付宝接口程序上传了(Alipay)。大家可以下载下来,解压后放到框架的Vendor目录中即可~

二、重新整理接口包文件,这一步应该算是比较关键的(个人认为)

下载下来的接口包文件有很多语言的源码,

我们选择 create_direct_pay_by_user-PHP-UTF-8 这个名称的接口文件。里面包括如下文件:

images文件里是支付宝相关的一些标志的图片,我们暂不管他,lib文件很重要,是整个接口的核心类文件;

alipay.config.php是相关参数的配置文件

alipayapi.php 是支付宝接口入口文件

notify_url.php 是服务器异步通知页面文件;

return_url.php 是页面跳转同步通知文件;

在ThinkPHP的框架文件下,找到Extend 进入,再进入Vendor,在Vendor文件夹下,新建文件夹Alipay,把支付宝作为第三方类库引入。然后,复制支付宝接口文件包中lib文件里的所有文件。一共4个文件,如下:

现在对以上文件进行重命名,

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");

为什么要去掉以上两个文件中的这两段代码,因为在项目中调用接口文件的时候,我把所有4个核心文件都通过vendor来进行引入。所以,这不再需要导入。

到此,支付宝接口包相关核心类库的整理基本完成。现在开始在项目中调用;

三、在项目中调用支付宝接口

调用分两步:

1、在配置文件中Conf/Config.php文件中对支付宝相关参数进行配置:

//支付宝配置参数

'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',

),

//以上配置项,是从接口包中alipay.config.php 文件中复制过来,进行配置;

'alipay'=>array(

//这里是卖家的支付宝账号,也就是你申请接口时注册的支付宝账号

'seller_email'=>'pay@xxx.com',

//这里是异步通知页面url,提交到项目的Pay控制器的notifyurl方法;

'notify_url'=>'http://www.xxx.com/Pay/notifyurl',

//这里是页面跳转通知url,提交到项目的Pay控制器的returnurl方法;

'return_url'=>'http://www.xxx.com/Pay/returnurl',

//支付成功跳转到的页面,我这里跳转到项目的User控制器,myorder方法,并传参payed(已支付列表)

'successpage'=>'User/myorder?ordtype=payed',

//支付失败跳转到的页面,我这里跳转到项目的User控制器,myorder方法,并传参unpay(未支付列表)

'errorpage'=>'User/myorder?ordtype=unpay',

),

2、新建一个PayAction控制器代码如下:

"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=newAlipaySubmit($alipay_config);

$html_text=$alipaySubmit->buildRequestForm($parameter,"post","确认");

echo $html_text;

}

/******************************

服务器异步通知页面方法

其实这里就是将notify_url.php文件中的代码复制过来进行处理

*******************************/

functionnotifyurl(){

/*

同理去掉以下两句代码;

*/

//require_once("alipay.config.php");

//require_once("lib/alipay_notify.class.php");

//这里还是通过C函数来读取配置项,赋值给$alipay_config

$alipay_config=C('alipay_config');

//计算得出通知验证结果

$alipayNotify=newAlipayNotify($alipay_config);

$verify_result=$alipayNotify->verifyNotify();

if($verify_result){

//验证成功

//获取支付宝的通知返回参数,可参考技术文档中服务器异步通知参数列表

$out_trade_no=$_POST['out_trade_no'];//商户订单号

$trade_no=$_POST['trade_no'];//支付宝交易号

$trade_status=$_POST['trade_status'];//交易状态

$total_fee=$_POST['total_fee'];//交易金额

$notify_id=$_POST['notify_id'];//通知校验ID。

$notify_time=$_POST['notify_time'];//通知的发送时间。格式为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,//交易状态

"notify_id"=>$notify_id,//通知校验ID。

"notify_time"=>$notify_time,//通知的发送时间。

"buyer_email"=>$buyer_email,//买家支付宝帐号;

);

if($_POST['trade_status']=='TRADE_FINISHED'){

//

}elseif($_POST['trade_status']=='TRADE_SUCCESS'){if(!checkorderstatus($out_trade_no)){

orderhandle($parameter);

//进行订单处理,并传送从支付宝返回的参数;

}

}

echo"success";//请不要修改或删除

}else{

//验证失败

echo"fail";

}

}

/*

页面跳转处理方法;

这里其实就是将return_url.php这个文件中的代码复制过来,进行处理;

*/

functionreturnurl(){

//头部的处理跟上面两个方法一样,这里不罗嗦了!

$alipay_config=C('alipay_config');

$alipayNotify=newAlipayNotify($alipay_config);//计算得出通知验证结果

$verify_result=$alipayNotify->verifyReturn();

if($verify_result){

//验证成功

//获取支付宝的通知返回参数,可参考技术文档中页面跳转同步通知参数列表

$out_trade_no=$_GET['out_trade_no'];//商户订单号

$trade_no=$_GET['trade_no'];//支付宝交易号

$trade_status=$_GET['trade_status'];//交易状态

$total_fee=$_GET['total_fee'];//交易金额

$notify_id=$_GET['notify_id'];//通知校验ID。

$notify_time=$_GET['notify_time'];//通知的发送时间。

$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,//交易状态

"notify_id"=>$notify_id,//通知校验ID。

"notify_time"=>$notify_time,//通知的发送时间。

"buyer_email"=>$buyer_email,//买家支付宝帐号

);

if($_GET['trade_status']=='TRADE_FINISHED'||$_GET['trade_status']=='TRADE_SUCCESS'){

if(!checkorderstatus($out_trade_no)){

orderhandle($parameter);//进行订单处理,并传送从支付宝返回的参数;

}

$this->redirect(C('alipay.successpage'));//跳转到配置项中配置的支付成功页面;

}else{

echo"trade_status=".$_GET['trade_status'];

$this->redirect(C('alipay.errorpage'));//跳转到配置项中配置的支付失败页面;

}

}else{

//验证失败

//如要调试,请看alipay_notify.php页面的verifyReturn函数

echo"支付失败!";

}

}

}

?>

3、这里有几个支付处理过程中需要用到的函数,我把这些函数写到了项目的Common/common.php中,这样不用手动调用,即可直接使用这些函数,代码如下:

//////////////////////////////////////////////////////

//Orderlist数据表,用于保存用户的购买订单记录;

/* Orderlist数据表结构;

CREATE TABLE `tb_orderlist` (

`id` int(11) NOT NULL AUTO_INCREMENT,

`userid` int(11) DEFAULT NULL,购买者userid

`username` varchar(255) DEFAULT NULL,购买者姓名

`ordid` varchar(255) DEFAULT NULL,订单号

`ordtime` int(11) DEFAULT NULL,订单时间

`productid` int(11) DEFAULT NULL,产品ID

`ordtitle` varchar(255) DEFAULT NULL,订单标题

`ordbuynum` int(11) DEFAULT '0',购买数量

`ordprice` float(10,2) DEFAULT '0.00',产品单价

`ordfee` float(10,2) DEFAULT '0.00',订单总金额

`ordstatus` int(11) DEFAULT '0',订单状态

`payment_type` varchar(255) DEFAULT NULL,支付类型

`payment_trade_no` varchar(255) DEFAULT NULL,支付接口交易号

`payment_trade_status` varchar(255) DEFAULT NULL,支付接口返回的交易状态

`payment_notify_id` varchar(255) DEFAULT NULL,

`payment_notify_time` varchar(255) DEFAULT NULL,

`payment_buyer_email` varchar(255) DEFAULT NULL,

`ordcode` varchar(255) DEFAULT NULL, //这个字段不需要的,大家看我西面的修正补充部分的说明!

`isused` int(11) DEFAULT '0',

`usetime` int(11) DEFAULT NULL,

`checkuser` int(11) DEFAULT NULL,

PRIMARY KEY (`id`)

) ENGINE=MyISAM AUTO_INCREMENT=5 DEFAULT CHARSET=utf8;

*/

//在线交易订单支付处理函数

//函数功能:根据支付接口传回的数据判断该订单是否已经支付成功;

//返回值:如果订单已经成功支付,返回true,否则返回false;

functioncheckorderstatus($ordid){

$Ord=M('Orderlist');

$ordstatus=$Ord->where('ordid='.$ordid)->getField('ordstatus');

if($ordstatus==1){

returntrue;

}else{

returnfalse;

}

}

//处理订单函数

//更新订单状态,写入订单支付后返回的数据

functionorderhandle($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);

}

/*-----------------------------------

2013.8.13更正

下面这个函数,其实不需要,大家可以把他删掉,

具体看我下面的修正补充部分的说明

------------------------------------*/

//获取一个随机且唯一的订单号;

functiongetordcode(){

$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;

}

}

四、总结几点

1、接口包中lib文件中的文件复制到Vendor后,重命名为TP规范的命名规则,为的是调用方便,当然你要改成其他名称也可以;

2、把执行支付操作(doalipay),处理异步返回结果(notifyurl),处理跳转返回结果(returnurl)三个支付接口的核心页面写到一个PayAction控制器中。

3、提交支付的页面中,可以在提交之前先把一些参数要传递的内容先通过隐藏域的方法组合好,比如金额先计算好,订单名称,订单描述等先用字符串组合好。然后提交表单,这样,在doalipay方法中只要直接构造传递参数,直接进行提交就行过了。

4、支付返回后的处理因为要在异步和跳转两个方法中都要进行相应的判断和处理,所以,把这些判断和处理写到一个自定义函数中,这样只要调用函数即可,使得代码更加清晰明了。

5、notify_url和return_url两种模式的返回url必须使用http://xxxxxxx这样的绝对路径,因为里是从支付宝平台返回到你的项目页面。不能使用相对路径。

以上代码在ThinkPHP3.0中正常使用!!

————————修正补充!!2013.08.13——————————

在第三部分中Orderlist数据表结构中,我有一个字段是OrdCode,这个字段是我系统中用来发送短信给客户的消费密码,也就是客户凭手机短信来消费时就要验证这个字段。

其实,大家在做系统的时候,可以把这个字段忽略,可以不用他。代码最后部分中,有一个获取一个随机且唯一的订单号的函数 getordcode(),这里我其实写错了,不是获取订单号,是ordcode,也就是消费密码,这个函数也不需要。系统中的订单号(ordid字段),我用的是时间戳。

在此修正!

——————–解决签名错误问题 修正 13-08-16————————

有人说在在调试时,签名出现无法通过的问题,产生问题的原因是在返回的URL地址中返回的参数中,可能存在__URL__这样的字符串。导致无法正确过滤参数。

解决办法:

方法1:

在向支付宝提交需要的参数时,不要使用__URL__,__PUBLIC__等TP中的模版替换变量,如果TP对这些变量解析不成功,会直接传递过去,所以,在这些地方直接使用原始的URL地址。

方法2:

在接口的Core文件中,加入改造后的过滤函数,如下:

/**

* 除去数组中的空值和签名参数

* @param $para 签名参数组

* return 去掉空值与签名参数后的新签名参数组

*/

functionparaFilter($para){

$para_filter=array();

while(list($key,$val)=each($para)){

if($key=="sign"||$key=="sign_type"||$key=='_URL_'||$val=="")continue;//添加了$key == '_URL_'

else$para_filter[$key]=$para[$key];

}

return$para_filter;

}

functionmyparaFilter($para){

$para_filter=array();

while(list($key,$val)=each($para)){

if($key=='_URL_')continue;

else$para_filter[$key]=$para[$key];

}

return$para_filter;

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容