php下应用微信支付功能

一、微信支付所包含的类别

  1. 付款码付款
  2. JSAPI支付
  3. Native支付
  4. APP支付
  5. H5支付
  6. 小程序支付
日常网页开发中付款码付款不常用;APP支付还没做研究;今天主要将剩下的几种方式
任何支付首先都需要预下单在微信平台做同一下单处理
github源码地址
二、统一下单
统一下单根据支付类型不同需要设置不同的 trade_type
1.以扫码支付统一下单为例
public function unifiedOrder($order_no,$total_fee) { // 当前时间 $time = time(); // 生成随机字符串 $nonceStr = md5($time.'#_pay@sign'); // API参数 $params = [ 'appid' => config('wxpay.appid'),//公众账户ID 'mch_id' => config('wxpay.mch_id'),//商户号 'attach' => '某某分店等信息',//附加数据 'body' => '腾讯充值中心-QQ会员充值',//商品描述 'nonce_str' => $nonceStr,//随机字符串 'sign' => 'MD5',//加密方式默认MD5可以省略 'notify_url' => base_url() . 'notice.php',// 异步通知地址,支付成功后微信支付异步回调地址 'out_trade_no' => $order_no,//商户订单号;如果有修改价格的需求建议表中单独设立微信支付编号;注意保证唯一性 'spbill_create_ip' => \request()->ip(),//终端IP 'total_fee' => $total_fee * 100, // 价格:单位分,默认币种人民币 'trade_type' => 'NATIVE', ]; // 生成签名 $params['sign'] = $this->makeSign($params); // 请求API $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder'; $result = $this->postXmlCurl($this->toXml($params), $url); //发出请求 $prepay = $this->fromXml($result); //格式转换// 请求失败; 只有返回数据return_code和result_code都为SUCCESS统一下单才成功 if ($prepay['return_code'] !== 'SUCCESS' || $prepay['result_code'] !== 'SUCCESS') { return false; }return $prepay; }

生成签名函数
private function makeSign($values) { //签名步骤一:按字典序排序参数 ksort($values); $string = $this->toUrlParams($values); //签名步骤二:在string后加入KEY $string = $string . '&key=' .config('wxpay.key'); //支付商户账户获取key //签名步骤三:MD5加密 $string = md5($string); //签名步骤四:所有字符转为大写 $result = strtoupper($string); return $result; }

格式化参数为url格式
private function toUrlParams($values) { $buff = ''; foreach ($values as $k => $v) { if ($k != 'sign' && $v != '' && !is_array($v)) { $buff .= $k . '=' . $v . '&'; } } return trim($buff, '&'); }

进行POST请求
private function postXmlCurl($xml, $url,$is_cert = false, $second = 30) { $ch = curl_init(); // 设置超时 curl_setopt($ch, CURLOPT_TIMEOUT, $second); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); //严格校验 // 设置header curl_setopt($ch, CURLOPT_HEADER, FALSE); // 要求结果为字符串且输出到屏幕上 curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE); // post提交方式 curl_setopt($ch, CURLOPT_POST, TRUE); curl_setopt($ch, CURLOPT_POSTFIELDS, $xml); if($is_cert){ curl_setopt($ch, CURLOPT_SSLCERTTYPE, 'PEM'); curl_setopt($ch, CURLOPT_SSLCERT, 'apiclient_cert.pem'); //证书的物理绝对路径 //默认格式为PEM,可以注释 curl_setopt($ch, CURLOPT_SSLKEYTYPE, 'PEM'); curl_setopt($ch, CURLOPT_SSLKEY, 'apiclient_key.pem'); //证书的物理绝对路径 }// 运行curl $data = https://www.it610.com/article/curl_exec($ch); curl_close($ch); return $data; }

将数据格式转换为xml
private function toXml($values) { if (!is_array($values) || count($values) <= 0 ) { return false; }$xml = ""; foreach ($values as $key => $val) { if (is_numeric($val)) { $xml .= "<" . $key . ">" . $val . ""; } else { $xml .= "<" . $key . ">"; } } $xml .= ""; return $xml; }

将xml格式转换为数组
private function fromXml($xml) { // 禁止引用外部xml实体 libxml_disable_entity_loader(true); return json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true); }

2.集成多项支付的统一下单
默认支付模式为NATIVE,再使用小程序支付时必须传入openid
如果下单失败返回false,成功返回微信支付后台返回值
public function unifiedOrder($wxpay_no,$total_fee,$trade_type='NATIVE',$openid='') { // 当前时间 $time = time(); // 生成随机字符串 $nonceStr = md5($time.'#_pay@sign'); // API参数 $params = [ 'appid' => config('wxpay.appid'),//公众账户ID 'mch_id' => config('wxpay.mch_id'),//商户号 'attach' => '某某分店等信息',//附加数据 'body' => '腾讯充值中心-QQ会员充值',//商品描述 'nonce_str' => $nonceStr,//随机字符串 'sign' => 'MD5',//加密方式默认MD5可以省略 'notify_url' => base_url() . 'notice.php',// 异步通知地址,支付成功后微信支付异步回调地址 'out_trade_no' => $wxpay_no,//商户订单号;如果有修改价格的需求建议表中单独设立微信支付编号;注意保证唯一性 'spbill_create_ip' => \request()->ip(),//终端IP 'total_fee' => $total_fee * 100, // 价格:单位分,默认币种人民币 'trade_type' => $trade_type, ]; //如果为JSAPI支付 if($trade_type == 'JSAPI'){ $params['openid'] = $openid; }// 生成签名 $params['sign'] = $this->makeSign($params); // 请求API $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder'; $result = $this->postXmlCurl($this->toXml($params), $url); //发出请求 $prepay = $this->fromXml($result); //格式转换// 请求失败; 只有返回数据return_code和result_code都为SUCCESS统一下单才成功 if ($prepay['return_code'] !== 'SUCCESS' || $prepay['result_code'] !== 'SUCCESS') { return false; }return $prepay; }

三、订单查询
传入微信支付单号
public function orderQuery($wspay_no) { // 当前时间 $time = time(); // 生成随机字符串 $nonceStr = md5($time.'#_pay@sign'); // API参数 $params = [ 'appid' => config('wxpay.appid'),//公众账户ID 'mch_id' => config('wxpay.mch_id'),//商户号 'nonce_str' => $nonceStr, 'out_trade_no' => $wspay_no, ]; // 生成签名 $params['sign'] = $this->makeSign($params); // 请求API $url = 'https://api.mch.weixin.qq.com/pay/orderquery'; $result = $this->postXmlCurl($this->toXml($params), $url); $prepay = $this->fromXml($result); return $prepay; }

四、关闭订单
订单生成后不能马上关闭,最短时间间隔为5分钟
public function closeOrder($wspay_no) { // 当前时间 $time = time(); // 生成随机字符串 $nonceStr = md5($time.'#_pay@sign'); // API参数 $params = [ 'appid' => config('wxpay.appid'),//公众账户ID 'mch_id' => config('wxpay.mch_id'),//商户号 'nonce_str' => $nonceStr, 'out_trade_no' => $wspay_no, ]; // 生成签名 $params['sign'] = $this->makeSign($params); // 请求API $url = 'https://api.mch.weixin.qq.com/pay/closeorder'; $result = $this->postXmlCurl($this->toXml($params), $url); $prepay = $this->fromXml($result); return $prepay; }

五、申请退款
1、交易时间超过一年的订单无法提交退款
2、微信支付退款支持单笔交易分多次退款,多次退款需要提交原支付订单的商户订单号和设置不同的退款单号。申请退款总金额不能超过订单金额。 一笔退款失败后重新提交,请不要更换退款单号,请使用原商户退款单号
3、请求频率限制:150qps,即每秒钟正常的申请退款请求次数不超过150次
错误或无效请求频率限制:6qps,即每秒钟异常或错误的退款申请请求不超过6次
4、每个支付订单的部分退款次数不能超过50次
5、此请求需要安全证书的认证,路径地址在postXmlCur()里配置
public function refund($wxpay_no,$out_refund_no,$total_fee,$refund_fee) { // 当前时间 $time = time(); // 生成随机字符串 $nonceStr = md5($time.'#_pay@sign'); // API参数 $params = [ 'appid' => 'wx6bf5eec027a0fe45',//公众账户ID 'mch_id' => '1511613151',//商户号 'nonce_str' => $nonceStr,//随机字符串 'out_trade_no' => $wxpay_no,//微信支付号 'out_refund_no' => $out_refund_no,//退款编号 'total_fee' => $total_fee * 100,//单号总金额 'refund_fee' => $refund_fee * 100,//退款金额,不能超过总金额 'notify_url' =>base_url() . 'refundNoticy',//退款成功通知地址 ]; // 生成签名 $params['sign'] = $this->makeSign($params); // 请求API $url = 'https://api.mch.weixin.qq.com/secapi/pay/refund'; $result = $this->postXmlCurl($this->toXml($params), $url,true); $prepay = $this->fromXml($result); return $prepay; }

六、请求支付的调用
  1. NATIVE模式(扫码支付)
根据返回数据code_url链接生成二维码供客户扫码识别
2.小程序支付(JSAPI)
需要再次签名
小程序调用支付窗口需要参数$nonceStr,$prepayId,$timeStamp以及加密签名后返回的paySign
private function makePaySign($nonceStr, $prepayId, $timeStamp) { $data = https://www.it610.com/article/['appId' => config('wxpay.app_id'),//公众账户ID 'nonceStr' => $nonceStr,//随机字符串,与调起支付时相同 'package' => 'prepay_id=' . $prepayId,//统一下单返回的prepay_id 'signType' => 'MD5', 'timeStamp' => $timeStamp,//时间戳,与调起支付时相同 ]; //签名步骤一:按字典序排序参数 ksort($data); $string = $this->toUrlParams($data); //签名步骤二:在string后加入KEY $string = $string . '&key=' . config('wxpay.key'); //签名步骤三:MD5加密 $string = md5($string); //签名步骤四:所有字符转为大写 $result = strtoupper($string); return $result; }

3.H5支付(MWEB)
【php下应用微信支付功能】根据返回的mweb_url调起微信进行支付

    推荐阅读