PHP微信支付结果通知与回调策略分析
程序员文章站
2023-11-30 20:23:16
本文实例讲述了php微信支付结果通知与回调策略。分享给大家供大家参考,具体如下:
支付完成后,微信会把相关支付结果和用户信息发送给商户,商户需要接收处理,并返回应答。...
本文实例讲述了php微信支付结果通知与回调策略。分享给大家供大家参考,具体如下:
支付完成后,微信会把相关支付结果和用户信息发送给商户,商户需要接收处理,并返回应答。
对后台通知交互时,如果微信收到商户的应答不是成功或超时,微信认为通知失败,微信会通过一定的策略定期重新发起通知,尽可能提高通知的成功率,但微信不保证通知最终能成功。 (通知频率为15/15/30/180/1800/1800/1800/1800/3600,单位:秒)
注意:同样的通知可能会多次发送给商户系统。商户系统必须能够正确处理重复的通知。
推荐的做法是,当收到通知进行处理时,首先检查对应业务数据的状态,判断该通知是否已经处理过,如果没有处理过再进行处理,如果处理过直接返回结果成功。在对业务数据进行状态检查和处理之前,要采用数据锁进行并发控制,以避免函数重入造成的数据混乱。
特别提醒:商户系统对于支付结果通知的内容一定要做签名验证,并校验返回的订单金额是否与商户侧的订单金额一致,防止数据泄漏导致出现“假通知”,造成资金损失。
$str='<xml><return_code><![cdata[success]]></return_code><return_msg><![cdata[ok]]></return_msg></xml>'; exit($str);
//微信支付回调 public function order_notice(){ $xml = $globals['http_raw_post_data']; //将服务器返回的xml数据转化为数组 $data = $this->fromxml($xml); // 保存微信服务器返回的签名sign $data_sign = $data['sign']; // sign不参与签名算法 unset($data['sign']); //$sign = self::makesign($data); $sign = $this->makesign($data); clog::setlog($data,'order_notice'); // 判断签名是否正确 判断支付状态 if ( ($sign===$data_sign) && ($data['return_code']=='success') && ($data['result_code']=='success') ) { //获取服务器返回的数据 $order_num = $data['out_trade_no']; //订单单号 $openid = $data['openid']; //付款人openid $total_fee = $data['total_fee']; //付款金额 $transaction_id = $data['transaction_id']; //微信支付流水号 $res = $this->order_notice_data_deal($order_num,$openid,$total_fee,$transaction_id); if (!$res) { $result = -2; } else { $result = 0; } }else{ $result = -1; } // 返回状态给微信服务器 if ($result == 0) { // 成功之后不会再回调 $str='<xml><return_code><![cdata[success]]></return_code><return_msg><![cdata[ok]]></return_msg></xml>'; } elseif ($result == -1){ // 失败后会继续发送几次回调 $str='<xml><return_code><![cdata[fail]]></return_code><return_msg><![cdata[签名失败]]></return_msg></xml>'; } elseif ($result == -2) { // 失败后会继续发送几次回调 $str='<xml><return_code><![cdata[fail]]></return_code><return_msg><![cdata[更改状态失败]]></return_msg></xml>'; } clog::setlog($result,'order_notice_result'); exit($str); }
附上微信支付助手函数
public function toxml($array){ if(!is_array($array)|| count($array) <= 0){ return ; } $xml = '<xml version="1.0">'; foreach ($array as $key=>$val){ if (is_numeric($val)){ $xml.="<".$key.">".$val."</".$key.">"; }else{ $xml.="<".$key."><![cdata[".$val."]]></".$key.">"; } } $xml.="</xml>"; return $xml; } public function fromxml($xml){ if(!$xml){ // 人工抛出错误 throw new exception("xml数据异常!"); } //将xml转为array //禁止引用外部xml实体 libxml_disable_entity_loader(true); $this->values = json_decode(json_encode(simplexml_load_string($xml, 'simplexmlelement', libxml_nocdata)), true); return $this->values; } public function makesign($data) { //签名步骤一:按字典序排序参数 ksort($data); $string = $this->tourlparams($data); //签名步骤二:在string后加入key $string = $string . "&key=".c('weixin_pay_key'); //签名步骤三:md5加密 $string = md5($string); //签名步骤四:所有字符转为大写 $result = strtoupper($string); return $result; } public function tourlparams($array) { $buff = ""; foreach ($array as $k => $v) { if($k != "sign" && $v != "" && !is_array($v)){ $buff .= $k . "=" . $v . "&"; } } $buff = trim($buff, "&"); return $buff; } // createnoncestr public function createnoncestr($length = 16) { $chars = "abcdefghijklmnopqrstuvwxyz0123456789"; $str =""; for ( $i = 0; $i < $length; $i++ ) { $str .= substr($chars, mt_rand(0, strlen($chars)-1), 1); } return $str; }
更多关于php相关内容感兴趣的读者可查看本站专题:《php微信开发技巧汇总》、《php编码与转码操作技巧汇总》、《php网络编程技巧总结》、《php字符串(string)用法总结》、《php中json格式数据操作技巧汇总》及《php针对xml文件操作技巧总结》
希望本文所述对大家php程序设计有所帮助。
上一篇: 浅析几种网站长尾关键词的选择技巧
下一篇: tornado捕获和处理404错误的方法