用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>"; } } */ ?> |