PHP开发APP端微信支付功能
用PHP开发APP端微信支付的一点个人心得
最近因为公司需求,要开发APP端上的微信支付,看了微信文档,感觉还不错,没有遇到太大的坑,需要注意的点不算太多。
写一个记事文档,作为备忘录。
APP支付流程
从上面的图片中,可以看出来,需要注意的流程是一共是3部分;
第一部分:调用下单API,返回预支付订单,签名之后再返回信息(4、5、6、7)
第二部分:异步通知(15、16)
第三部分:最后的判断支付结果
最需要注意的就是第一部分:调用下单API,返回预支付订单,签名之后再返回信息
微信文档中有详细的说明,这里不再赘述。
附录一下我的代码,伸手党,稍微改点代码就可以用了。
代码如下 | 复制代码 |
//入口函数 functionweChatPay(){ $json=array(); //生成预支付交易单的必选参数: $newPara=array(); //应用ID $newPara["appid"] ="wx2421b1c4370ec43b"; //商户号 $newPara["mch_id"] ="10000100"; //设备号 $newPara["device_info"] ="WEB"; //随机字符串,这里推荐使用函数生成 $newPara["nonce_str"] ="1add1a30ac87aa2db72f57a2375d8fec"; //商品描述 $newPara["body"] ="APP支付测试"; //商户订单号,这里是商户自己的内部的订单号 $newPara["out_trade_no"] ="1415659990"; //总金额 $newPara["total_fee"] = 1; //终端IP $newPara["spbill_create_ip"] =$_SERVER["REMOTE_ADDR"]; //通知地址,注意,这里的url里面不要加参数 $newPara["notify_url"] ="http://wxpay.wxutil.com/pub_v2/pay/notify.v2.php"; //交易类型 $newPara["trade_type"] ="APP"; //第一次签名 $newPara["sign"] = produceWeChatSign($newPara); //把数组转化成xml格式 $xmlData= getWeChatXML($newPara); //利用PHP的CURL包,将数据传给微信统一下单接口,返回正常的prepay_id $get_data= sendPrePayCurl($xmlData); //返回的结果进行判断。 if($get_data['return_code'] =="SUCCESS"&&$get_data['result_code'] =="SUCCESS"){ //根据微信支付返回的结果进行二次签名 //二次签名所需的随机字符串 $newPara["nonce_str"] ="5K8264ILTKCH16CQ2502SI8ZNMTM67VS"; //二次签名所需的时间戳 $newPara['timeStamp'] = time().""; //二次签名剩余参数的补充 $secondSignArray=array( "appid"=>$newPara['appid'], "noncestr"=>$newPara['nonce_str'], "package"=>"Sign=WXPay", "prepayid"=>$get_data['prepay_id'], "partnerid"=>$newPara['mch_id'], "timestamp"=>$newPara['timeStamp'], ); $json['datas'] =$secondSignArray; $json['ordersn'] =$newPara["out_trade_no"]; $json['datas']['sign'] = weChatSecondSign($newPara,$get_data['prepay_id']); $json['message'] ="预支付完成"; //预支付完成,在下方进行自己内部的业务逻辑 /*****************************/ returnjson_encode($json); } else{ $json['message'] =$get_data['return_msg']; } } returnjson_encode($json); }
//第一次签名的函数produceWeChatSign functionproduceWeChatSign($newPara){ $stringA= self::getSignContent($newPara); $stringSignTemp=$stringA."&key=192006250b4c09247ec02edce69f6a2d"; returnstrtoupper(MD5($stringSignTemp)); }
//生成xml格式的函数 publicstaticfunctiongetWeChatXML($newPara){ $xmlData=" foreach($newParaas$key=>$value) { $xmlData=$xmlData."<".$key.">".$value.""; } $xmlData=$xmlData.""; return$xmlData; }
//通过curl发送数据给微信接口的函数 functionsendPrePayCurl($xmlData) { $url="https://api.mch.weixin.qq.com/pay/unifiedorder"; $header[] ="Content-type: text/xml"; $curl= curl_init(); curl_setopt($curl, CURLOPT_HTTPHEADER,$header); curl_setopt($curl, CURLOPT_URL,$url); curl_setopt($curl, CURLOPT_RETURNTRANSFER, true); curl_setopt($curl, CURLOPT_POST, 1); curl_setopt($curl, CURLOPT_POSTFIELDS,$xmlData); $data= curl_exec($curl); if(curl_errno($curl)) { printcurl_error($curl); } curl_close($curl); returnself::XMLDataParse($data);
}
//xml格式数据解析函数 publicstaticfunctionXMLDataParse($data){ $msg=array(); $msg= (array)simplexml_load_string($data,'SimpleXMLElement', LIBXML_NOCDATA); return$msg; }
//二次签名的函数 functionweChatSecondSign($newPara,$prepay_id){ $secondSignArray=array( "appid"=>$newPara['appid'], "noncestr"=>$newPara['nonce_str'], "package"=>"Sign=WXPay", "prepayid"=>$prepay_id, "partnerid"=>$newPara['mch_id'], "timestamp"=>$newPara['timeStamp'], ); $stringA= self::getSignContent($secondSignArray); $stringSignTemp=$stringA."&key=192006250b4c09247ec02edce69f6a2d"; returnstrtoupper(MD5($stringSignTemp)); } |
两个注意点:
1.二次签名需要在后台完成,并且完成之后,连带着二次签名所用的所有信息一起传给前端,让前段唤起微信支付。这样不容易出现没法吊起微信支付的情况。
2.两次签名,用的是不同的随机字符串。
/**
*
* 中英混合字符串长度判断
* @param unknown_type $str
* @param unknown_type $charset
*/
functionstrLength($str,$charset='utf-8') {
if($charset=='utf-8')
$str= iconv ('utf-8','gb2312',$str);
$num=strlen($str);
$cnNum= 0;
for($i= 0;$i<$num;$i++) {
if(ord (substr($str,$i+ 1, 1 ) ) > 127) {
$cnNum++;
$i++;
}
}
$enNum=$num- ($cnNum* 2);
$number= ($enNum/ 2) +$cnNum;
returnceil($number);
}
/**
*
* 中英混合的字符串截取
* @param unknown_type $sourcestr
* @param unknown_type $cutlength
*/
functioncut_str($sourcestr,$cutlength) {
$returnstr=''
$i= 0;
$n= 0;
$str_length=strlen($sourcestr);//字符串的字节数
while( ($n<$cutlength)and($i<=$str_length) ) {
$temp_str=substr($sourcestr,$i, 1 );
$ascnum= Ord ($temp_str);//得到字符串中第$i位字符的ascii码
if($ascnum>= 224)//如果ASCII位高与224,
{
$returnstr=$returnstr.substr($sourcestr,$i, 3 );//根据UTF-8编码规范,将3个连续的字符计为单个字符
$i=$i+ 3;//实际Byte计为3
$n++;//字串长度计1
}elseif($ascnum>= 192)//如果ASCII位高与192,
{
$returnstr=$returnstr.substr($sourcestr,$i, 2 );//根据UTF-8编码规范,将2个连续的字符计为单个字符
$i=$i+ 2;//实际Byte计为2
$n++;//字串长度计1
}elseif($ascnum>= 65 &&$ascnum<= 90)//如果是大写字母,
{
$returnstr=$returnstr.substr($sourcestr,$i, 1 );
$i=$i+ 1;//实际的Byte数仍计1个
$n++;//但考虑整体美观,大写字母计成一个高位字符
}else//其他情况下,包括小写字母和半角标点符号,
{
$returnstr=$returnstr.substr($sourcestr,$i, 1 );
$i=$i+ 1;//实际的Byte数计1个
$n=$n+ 0.5;//小写字母和半角标点等与半个高位字符宽...
}
}
if($str_length>$cutlength) {
$returnstr=$returnstr."...";//超过长度时在尾处加上省略号
}
return$returnstr;
}
代码如下 | 复制代码 |
利用嵌套数组 拼接混合json -包含对象数组
代码如下 | 复制代码 |
<?php
// 自 PHP 5.4 起 $array= [ "status"=>"0", "message"=>"ok", "arr"=> [] ];
classPerson { public$name; public$age;
//定义一个构造方法初始化赋值 publicfunction__construct($name,$age) { $this->name=$name; $this->age=$age; } }
for($i=0;$i<10;$i++) { $p=newPerson("ren",$i); $array["arr"][]=$p; }
//var_dump($array);
echojson_encode($array);
?> |
php利用嵌套数组 解析混合json 包含对象数组
代码如下 | 复制代码 |
<?php functionjson_to_array($web) { $arr=array(); foreach($webas$k=>$v) { if(is_object($v))$arr[$k]=json_to_array($v);//判断类型是不是object else$arr[$k]=$v; } return$arr; } $s='{"webname":"homehf","url":"www.homehf.com","qq":"744348666"}' //将字符转成JSON $web=json_decode($s); $arr=array(); foreach($webas$k=>$v) $arr[$k]=$v; echo" "; print_r($arr); echo"";
$s='{"webname":"homehf","url":"www.homehf.com","contact":{"qq":"744348666","mail":"nieweihf@163.com","xx":"xxxxxxx"}}' $web=json_decode($s); $arr=json_to_array($web); echo" "; print_r($arr); echo"";
/************************************************************************ ************************************************************************/ $s='{"webname":"homehf","url":"www.homehf.com","contact":{"qq":"744348666","mail":"nieweihf@163.com","xx":"xxxxxxx"}}' $web=json_decode($s); echo'网站名称:'.$web->webname.' echo' /************************************************************************ ************************************************************************/ $s='{"webname":"homehf","url":"www.homehf.com","contact":{"qq":"744348666","mail":"nieweihf@163.com","xx":"xxxxxxx"}}' $web=json_decode($s); echojson_encode($web);
$mys='{"status":"0","message":"ok","arr":[{"name":"ren","age":0},{"name":"ren","age":1},{"name":"ren","age":2}, {"name":"ren","age":3},{"name":"ren","age":4},{"name":"ren","age":5},{"name":"ren","age":6},{"name":"ren","age":7}, {"name":"ren","age":8},{"name":"ren","age":9}]}'
$myweb=json_decode($mys);
echo$myweb->status;
for($i=0;$i<10;$i++) { echo$myweb->arr[$i]->age; echo' } ?> |
代码如下 | 复制代码 |
<?php /**//* * @(#)UploadFile.php * * 可同时处理用户多个上传文件。效验文件有效性后存储至指定目录。 * 可返回上传文件的相关有用信息供其它程序使用。(如文件名、类型、大小、保存路径) * 使用方法请见本类底部(UploadFile类使用注释)信息。 * */ classUploadFile { var$user_post_file=array();//用户上传的文件 var$save_file_path; //存放用户上传文件的路径 var$max_file_size; //文件最大尺寸 var$last_error; //记录最后一次出错信息 //默认允许用户上传的文件类型 var$allow_type=array('gif','jpg','png','zip','rar','txt','doc','pdf'); var$final_file_path;//最终保存的文件名 var$save_info=array();//返回一组有用信息,用于提示用户。 /**//** * 构造函数,用与初始化相关信息,用户待上传文件、存储路径等 * * @param Array $file 用户上传的文件 * @param String $path 存储用户上传文件的路径 * @param Integer $size 允许用户上传文件的大小(字节) * @param Array $type 此数组中存放允计用户上传的文件类型 */ functionUploadFile($file,$path,$size= 2097152,$type='') { $this->user_post_file =$file; $this->save_file_path =$path; $this->max_file_size =$size;//如果用户不填写文件大小,则默认为2M. if($type!='') $this->allow_type =$type; } /**//** * 存储用户上传文件,检验合法性通过后,存储至指定位置。 * @access public * @return int 值为0时上传失败,非0表示上传成功的个数。 */ functionupload() { for($i= 0;$i<count($this->user_post_file['name']);$i++) { //如果当前文件上传功能,则执行下一步。 if($this->user_post_file['error'][$i] == 0) { //取当前文件名、临时文件名、大小、扩展名,后面将用到。 $name=$this->user_post_file['name'][$i]; $tmpname=$this->user_post_file['tmp_name'][$i]; $size=$this->user_post_file['size'][$i]; $mime_type=$this->user_post_file['type'][$i]; $type=$this->getFileExt($this->user_post_file['name'][$i]); //检测当前上传文件大小是否合法。 if(!$this->checkSize($size)) { $this->last_error ="The file size is too big. File name is: ".$name; $this->halt($this->last_error); continue; } //检测当前上传文件扩展名是否合法。 if(!$this->checkType($type)) { $this->last_error ="Unallowable file type: .".$type." File name is: ".$name; $this->halt($this->last_error); continue; } //检测当前上传文件是否非法提交。 if(!is_uploaded_file($tmpname)) { $this->last_error ="Invalid post file method. File name is: ".$name; $this->halt($this->last_error); continue; } //移动文件后,重命名文件用。 $basename=$this->getBaseName($name,".".$type); //移动后的文件名 $saveas=$basename."-".time().".".$type; //组合新文件名再存到指定目录下,格式:存储路径 + 文件名 + 时间 + 扩展名 $this->final_file_path =$this->save_file_path."/".$saveas; if(!move_uploaded_file($tmpname,$this->final_file_path)) { $this->last_error =$this->user_post_file['error'][$i]; $this->halt($this->last_error); continue; } //存储当前文件的有关信息,以便其它程序调用。 $this->save_info[] =array("name"=>$name,"type"=>$type, "mime_type"=>$mime_type, "size"=>$size,"saveas"=>$saveas, "path"=>$this->final_file_path); } } returncount($this->save_info);//返回上传成功的文件数目 } /**//** * 返回一些有用的信息,以便用于其它地方。 * @access public * @return Array 返回最终保存的路径 */ functiongetSaveInfo() { return$this->save_info; } /**//** * 检测用户提交文件大小是否合法 * @param Integer $size 用户上传文件的大小 * @access private * @return boolean 如果为true说明大小合法,反之不合法 */ functioncheckSize($size) { if($size>$this->max_file_size) { returnfalse; } else{ returntrue; } } /**//** * 检测用户提交文件类型是否合法 * @access private * @return boolean 如果为true说明类型合法,反之不合法 */ functioncheckType($extension) { foreach($this->allow_typeas$type) { if(strcasecmp($extension,$type) == 0) returntrue; } returnfalse; } /**//** * 显示出错信息 * @param $msg 要显示的出错信息 * @access private */ functionhalt($msg) { printf("<b><UploadFile Error:></b> %s <br>\n",$msg); } /**//** * 取文件扩展名 * @param String $filename 给定要取扩展名的文件 * @access private * @return String 返回给定文件扩展名 */ functiongetFileExt($filename) { $stuff=pathinfo($filename); return$stuff['extension']; } /**//** * 取给定文件文件名,不包括扩展名。 * eg: getBaseName("j:/hexuzhong.jpg"); //返回 hexuzhong * * @param String $filename 给定要取文件名的文件 * @access private * @return String 返回文件名 */ functiongetBaseName($filename,$type) { $basename=basename($filename,$type); return$basename; } } /**//******************** UploadFile类使用注释 //注意,上传组件name属性不管是一个还是多个都要使用数组形式,如: <input type="file" name="user_upload_file[]"> <input type="file" name="user_upload_file[]"> //如果用户点击了上传按钮。 if ($_POST['action'] == "上传") { //设置允许用户上传的文件类型。 $type = array('gif', 'jpg', 'png', 'zip', 'rar'); //实例化上传类,第一个参数为用户上传的文件组、第二个参数为存储路径、 //第三个参数为文件最大大小。如果不填则默认为2M //第四个参数为充许用户上传的类型数组。如果不填则默认为gif, jpg, png, zip, rar, txt, doc, pdf $upload = new UploadFile($_FILES['user_upload_file'], 'j:/tmp', 100000, $type); //上传用户文件,返回int值,为上传成功的文件个数。 $num = $upload->upload(); if ($num != 0) { echo "上传成功<br>"; //取得文件的有关信息,文件名、类型、大小、路径。用print_r()打印出来。 print_r($upload->getSaveInfo()); //格式为: Array // ( // [0] => Array( // [name] => example.txt // [type] => txt // [size] => 526 // [path] => j:/tmp/example-1108898806.txt // ) // ) echo $num."个文件上传成功"; } else { echo "上传失败<br>"; } } */ ?> |
相关文章
- 微信支付,即便交了保证金,你还是处理测试阶段,不能正式发布。必须到你通过程序测试提交订单、发货通知等数据到微信的系统中,才能申请发布。然后,因为在微信中是通过JS方式调用API,必须在微信后台设置支付授权目录,而且要到...2014-05-31
- 为公司系统业务需要,这几天了解了一下微信和支付宝扫码支付的接口,并用c#实现了微信和支付宝扫码支付的功能。需要的朋友跟随小编一起看看吧...2020-06-25
- 这篇文章主要介绍了C#实现图片放大功能的按照像素放大图像方法,功能非常实用,需要的朋友可以参考下...2020-06-25
- 在本篇文章中小编给各位整理了一篇关于python中翻译功能translate模块实现方法,有需要的朋友们可以参考下。...2020-12-18
- 这篇文章主要给大家介绍了关于微信小程序实现导航功能的操作步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-03-10
- EMUI11值得升级吗?好不好用?下面小编带来EMUI11上手体验,一起来看看手机鸿蒙OS的提前预演...2020-12-08
- AngularJS 让人爱不释手的八种功能,想知道AngularJS哪八种功能让人喜欢就快点看下本文吧...2016-03-28
- phpmyadmin还有高级功能可能大部份站长不知道吧,今天本文章就来给大家介绍phpMyAdmin 高级功能设置的方法图解,希望文章对大家会有所帮助。 phpMyAdmin 安装后,默认...2016-11-25
- 这篇文章主要介绍了iOS APP实现微信H5支付示例总结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-06-30
- 这篇文章主要为大家分享了一份很全面的JavaScript常用功能汇总集合,一些常用的额JS 对象、基本数据结构、功能函数等,感兴趣的小伙伴们可以参考一下...2016-01-24
- 小爱同学5.0即将发布,据已知报道小爱同学5.0将新增机型,跟着小编一起来看看吧,顺便了解下即将都有哪些新功能面市吧...2020-12-08
Night Shift是什么意思 Night Shift有什么功能及作用?
Night Shift是IOS9.3正式版系统新增加的功能之一,很多伙伴们都不清楚Night Shift是什么意思?以及Night Shift有什么用途?对此,本文小编就为大家详细介绍Night Shift的含义及作用...2016-07-04- 这篇文章主要为大家详细介绍了微信小程序支付C#后端源码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-06-25
- 这篇文章主要为大家详细介绍了javaScript实现支付10秒倒计时,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-10-28
Mate30EPro有哪些功能特色 Mate30EPro功能特色介绍
Mate30EPro刚刚发布,那么这款华为新机Mate30EPro有哪些功能特色呢,一起来了解一下吧...2020-12-08- <?php include_once("mime.inc"); $mm = new MIME(); $to = "customer@263.net"; $subject = $mm->encode("商城","gb2312"); // 编码 $msg = "注册会员成功<br>"; $m...2016-11-25
- 最近开发微信小程序进入到支付阶段,一直以来从事App开发,所以支付流程还是熟记于心的。下面通过本文给大家讲述下微信小程序进行微信支付的步骤,需要的朋友可以参考下...2016-12-02
php基于ob_start(ob_gzhandler)实现网页压缩功能的方法
小编推荐的这篇文章介绍了php基于ob_start(ob_gzhandler)实现网页压缩功能的方法,非常实用,有兴趣的同学可以看看。 PHP生成网页后传送给浏览器显示 ,页面的打开速度...2017-07-06- 小米在最新的MIUI8中正式加入了录屏功能,用户再也不需要ROOT手机,然后借助“录屏大师”等APP来录屏了,借助系统自带的录屏功能即可轻松录制屏幕。那么,MIUI8录屏功能录屏功能在哪?MIUI8录屏功能怎么使用?本文将提供MIUI小米手机录屏使用方法图文教程供大家了解...2017-07-06
- 环境:中文w2k+apache+php4.06(smtp在linux上) 我在使用mail函数发信的时候发现,如果我的subject是中文,那么收到的mail就会把中文显示为"XXXX" 调用过程如下: <?php $t...2016-11-25