联系QQ 284710375
首页 > 技术分享 > 原生PHP
收藏

PHP实现APP微信支付2019-04-10 18:00:00

大潇博客 原创文章,转载请标明出处

一丶 PHP后台生成预支付交易单,返回正确的预支付交易回话标识后再在APP里面调起支付!官方文档:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_1


通俗的讲,微信支付有以下步骤:

1、用户在APP服务端生成预支付订单

2、将预支付订单信息,结合其它参数,发送给微信

3、微信接收参数后,验证是否合法

(服务端请求微信支付接口后将接收微信返回的参数)

4、根据微信返回的参数,APP将验证参数合法性,若合法,调起微信支付界面

5、用户支付成功,微信将请求APP服务端的微信支付回调接口

6、APP服务端微信支付回调接口,接收微信返回的参数,根据参数处理支付结果(成功or失败)

根据文档拼接微信需要的参数,这里需要几个方法,直接上代码!


微信验证成功,生成订单并返回如下参数:

生成订单失败的参数,可参考博客内H5支付,参数格式相同。


支付成功,微信请求回调时的参数


1、传输给微信的参数要组装成xml格式发送,传如参数数组!

function ToXml($data=array()){

if(!is_array($data) || count($data) <= 0){

      return '数组异常';

}

$xml = "<"."xml".">";

foreach($data as $key=>$val){

      if(is_numeric($val)){

            $xml.="<".$key.">".$val."<".'/'.$key.">";

      }else{

            $xml.="<".$key."><"."![cdata[".$val."]]><".'/'.$key.">";

      }

}

$xml.="<".'/'."xml>";

return $xml;

}


2、生成随机字符串,微信所需参数!这里方法很多,看自己爱好都行!

function rand_code(){

$str = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';//62个字符

$str = str_shuffle($str);

$str = substr($str,0,32);

return $str;

}


3、这里是微信比较重要的一步了,这个方法会多次用到!生成签名

private function getSign($params){

ksort($params);        //将参数数组按照参数名ASCII码从小到大排序

foreach ($params as $k => $item) {

      if (!empty($item)) {         //剔除参数值为空的参数

            $newArr[] = $k.'='.$item;     // 整合新的参数数组

      }

}

$stringA = implode("&", $newArr);         //使用 & 符号连接参数

$stringSignTemp = $stringA."&key="."************************";        //拼接key

//key是在商户平台API安全里自己设置的,设置路径:微信商户平台(pay.weixin.qq.com)-->账户设置-->API安全-->密钥设置

$stringSignTemp = MD5($stringSignTemp);       //将字符串进行MD5加密

$sign = strtoupper($stringSignTemp);      //将所有字符转换为大写

return $sign;

}


4、传递参数给微信,生成预支付订单! 接收微信返回的数据,在反给APP端,APP端调用支付接口,完成支付!APP端所需参数见微信文档:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_12&index=2

public function wx_pay() {

$nonce_str = $this->rand_code();        //调用随机字符串生成方法获取随机字符串

$data['appid'] ='wxdbc5dc*******';   //appid

$data['mch_id'] = '1493*****' ;        //商户号

$data['body'] = "APP支付测试";

$data['spbill_create_ip'] = $_SERVER['HTTP_HOST'];   //ip地址

$data['total_fee'] = 1;                         //金额

$data['out_trade_no'] = time().mt_rand(10000,99999);    //商户订单号,不能重复

$data['nonce_str'] = $nonce_str;                   //随机字符串

$data['notify_url'] = 'http://xxx.xxx.com/wx_notify';   //回调地址,用户接收支付后的通知,必须为能直接访问的网址,不能跟参数

$data['trade_type'] = 'APP';      //支付方式

//将参与签名的数据保存到数组  注意:以上几个参数是追加到$data中的,$data中应该同时包含开发文档中要求必填的剔除sign以外的所有数据

$data['sign'] = $this->getSign($data);        //获取签名

$xml = $this->ToXml($data);            //数组转xml

//curl 传递给微信方

$url = "https://api.mch.weixin.qq.com/pay/unifiedorder";

//header("Content-type:text/xml");

$ch = curl_init();

curl_setopt($ch,CURLOPT_URL, $url);

if(stripos($url,"https://")!==FALSE){

      curl_setopt($ch, CURLOPT_SSLVERSION, CURL_SSLVERSION_TLSv1);

      curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);

      curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE);

} else {

      curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,TRUE);

      curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,2);//严格校验

}

//设置header

curl_setopt($ch, CURLOPT_SSLVERSION, CURL_SSLVERSION_TLSv1);

curl_setopt($ch, CURLOPT_HEADER, FALSE);

//要求结果为字符串且输出到屏幕上

curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);

//设置超时

curl_setopt($ch, CURLOPT_TIMEOUT, 30);

curl_setopt($ch, CURLOPT_POST, TRUE);

//传输文件

curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);

//运行curl

$data = curl_exec($ch);

//返回结果

if($data){

      curl_close($ch);

      //返回成功,将xml数据转换为数组.

      $re = $this->FromXml($data);

      if($re['return_code'] != 'SUCCESS'){

            json("201",'签名失败');

      }else{

            //接收微信返回的数据,传给APP!

            $arr =array(

                  'prepayid' =>$re['prepay_id'],

                  'appid' => 'wxdbc5dc*****',

                  'partnerid' => '14937****',

                  'package' => 'Sign=WXPay',

                  'noncestr' => $nonce_str,

                  'timestamp' =>time(),

            );

            //第二次生成签名

            $sign = $this->getSign($arr);

            $arr['sign'] = $sign;

            json('200','签名成功',$arr);

      }

} else {

      $error = curl_errno($ch);

      curl_close($ch);

      json('201',"curl出错,错误码:$error");

}

}


5、将xml数据转换为数组,接收微信返回数据时用到

function FromXml($xml){

if(!$xml){

      echo "xml数据异常!";

}

//将XML转为array

//禁止引用外部xml实体

libxml_disable_entity_loader(true);

$data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);

return $data;

}


二、APP支付成功后,会调用你填写的回调地址。返回参数详见微信文档:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_7&index=3

// 微信支付回调

function wx_notify(){

//接收微信返回的数据数据,返回的xml格式

$xmlData = file_get_contents('php://input');

//将xml格式转换为数组

$data = $this->FromXml($xmlData);

//用日志记录检查数据是否接受成功,验证成功一次之后,可删除。

$file = fopen('./log.txt', 'a+');

fwrite($file,var_export($data,true));

//为了防止假数据,验证签名是否和返回的一样。

//记录一下,返回回来的签名,生成签名的时候,必须剔除sign字段。

$sign = $data['sign'];

unset($data['sign']);

if($sign == $this->getSign($data)){

      //签名验证成功后,判断返回微信返回的

      if ($data['result_code'] == 'SUCCESS') {

            //根据返回的订单号做业务逻辑

            $arr = array(

                  'pay_status' => 1,

            );

            $re = M('order')->where(['order_sn'=>$data['out_trade_no']])->save($arr);

            //处理完成之后,告诉微信成功结果!

            if($re){

                        echo '<'.'xml'.'>'.

                        '<'.'return_code'.'><'.'!'."[cdata[success]]><".'/'."return_code>".

                 '<'.'return_msg'.'><'.'!'."[cdata[ok]]><".'/'."return_msg>".

                        "<".'/'."xml>";

                 exit();

            }

      else { //支付失败,输出错误信息

            $file = fopen('./log.txt', 'a+');

            fwrite($file,"错误信息:".$data['return_msg'].date("Y-m-d H:i:s"),time().'\'.'r'.'\'.'n');    

      }

}else{

      $file = fopen('./log.txt', 'a+');

      fwrite($file,"错误信息:签名验证失败".date("Y-m-d H:i:s"),time().'\'.'r'.'\'.'n'); 

}

}

在这里,微信APP支付流程就成功走完了!

由于富文本可能对特殊字符过滤,下面是所有代码的图片,并附带源文件

shield.png

下面为代码源文件(请用电脑下载):

打赏

阅读排行

大家都在搜

博客维护不易,感谢你的肯定
扫码打赏,建议金额1-10元
  • 15601023311