前言:这里我就假装你已经注册了微信小程序,并且基本的配置都已经好了。注: 个人注册小程序不支持微信支付,所以我还是假装你是企业或者个体工商户的微信小程序,其他的商户号注册,二者绑定,授权,支付开通,就阅读文档吧,这里我先负责实战。

微信小程序支付开发文档

https://pay.weixin.qq.com/wiki/doc/api/wxa/wxa_api.php?chapter=7_3&index=1

基本流程:

1. 申请商户平台账号 https://pay.weixin.qq.com/index.php/core/home/login?return_url=%2F

2. 微信小程序绑定已有商户号并开通微信支付 http://kf.qq.com/faq/140225MveaUz161230yqiIby.html

3. 登录商户平台对小程序授权,下载支付证书,记录商户号,支付密钥。

4. 阅读微信支付官方文档,完成接口的对接编码。

开发支付流程: 

1. 微信小程序的基本配置。(app_id[小程序唯一id],mch_id[商户号],md5_key[支付密钥],notify_url[异步回调通知] )。

2. 按微信要求的顺序将参数组成键值对数组,并对其进行签名(先将参数进行字段排序,参数可以处理中文字符,在请求参数字符串后拼上支付密钥,最后md5,签名完成)

3. 所有请求参数和签名一起组成新数组,再转为XML。

4. 以XML格式参数,POST请求方式对https://api.mch.weixin.qq.com/pay/unifiedorder发起统一下单请求。

5. 微信服务器接收下单请求,返回预支付ID(prepay_id)到自己服务端。

6. 自己服务端联合预支付ID,小程序APPID,32位随机串,时间戳,签名方式一并返回到小程序。

7. 小程序根据微信提供的函数和返回的参数集调起微信支付。

8. 支付完成,微信通过异步通知到自己服务指定的控制器。

9. 接受微信返回的通知,将XML转为数组,需要先判断通知过来的是不是同一个订单(根据订单号),因为有时微信异步通知,自己服务器未接收处理,他会过一段时间重复发起通知。

10. 根据通知状态,更新自己业务的数据表,最后返回一个成功标识的XML给微信服务器。

一、支付配置

  1. 'wxxcx' =>[
  2. 'app_id' => 'wx4c0e*******664b4', // 微信小程序appid
  3. 'mch_id' => '149***3342', // 微信商户id
  4. 'md5_key' => '3FN8WHO**********iPnNoJ576AxMmwQ', // 微信支付密钥
  5. 'app_cert_pem' => APP_PATH.'v1/wechat_cert/apiclient_cert.pem', // 支付证书,统一下单不需,退款等其他接口需要
  6. 'app_key_pem' => APP_PATH.'v1/wechat_cert/apiclient_key.pem',
  7. 'sign_type' => 'MD5',// MD5 HMAC-SHA256
  8. 'limit_pay' => [
  9. ],
  10. 'fee_type' => 'CNY',// 货币类型 当前仅支持该字段
  11. 'notify_url' => 'https://***********.com/v1/Pay/notifyUrlApi', // 异步通知地址
  12. 'redirect_url' => '',
  13. 'return_raw' => false,
  14. ]

二、前端传来的参数或服务端生成
$this->openid = $openid;      // 前端也可不传
 $this->out_trade_no = $out_trade_no;   // 服务端生成
$this->body = $body;
$this->total_fee = $total_fee;    // 最好服务端数据库抓取,避免前端传
$this->spbill_create_ip = $spbill_create_ip;  // 请求的ip地址

三、封装统一下单类

  1. <?php
  2. /**
  3. * @author: fuchao
  4. * @createTime: 2018-04-30 18:02
  5. * @description: 小程序微信支付
  6. * 公众号:ZEROFC_DEV
  7. */
  8. namespace app\v1\extend;
  9. class WeixinPay {
  10. protected $appid;
  11. protected $mch_id;
  12. protected $key;
  13. protected $openid;
  14. protected $out_trade_no;
  15. protected $body;
  16. protected $total_fee;
  17. protected $notify_url;
  18. protected $spbill_create_ip;
  19. function __construct($appid, $openid, $mch_id, $key,$out_trade_no,$body,$total_fee,$notify_url,$spbill_create_ip) {
  20. $this->appid = $appid;
  21. $this->openid = $openid;
  22. $this->mch_id = $mch_id;
  23. $this->key = $key;
  24. $this->out_trade_no = $out_trade_no;
  25. $this->body = $body;
  26. $this->total_fee = $total_fee;
  27. $this->notify_url = $notify_url;
  28. $this->spbill_create_ip = $spbill_create_ip;
  29. }
  30. /************测试方法可删除*****************/
  31. public function test() {
  32. $ha = "hello world";
  33. return $this->appid;
  34. }
  35. /************可删除*****************/
  36. public function pay() {
  37. // var_dump($this->notify_url);
  38. // die;
  39. //统一下单接口
  40. $return = $this->weixinapp();
  41. return $return;
  42. }
  43. //统一下单接口
  44. private function unifiedorder() {
  45. $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
  46. // 这里的参数顺序一定要按下面的,不然可能就一直报商户号此功能未授权等错误
  47. $parameters = array(
  48. 'appid' => $this->appid, // 小程序ID
  49. //'body' => 'test', // 商品描述
  50. 'body' => $this->body,
  51. 'mch_id' => $this->mch_id, // 商户号
  52. 'nonce_str' => $this->createNoncestr(), // 随机字符串
  53. 'notify_url' => $this->notify_url, //'https://shop.gdpress.cn/syw_jingzhun/index.php/Api/xiaochengxu/notify_url_api', // 通知地址 确保外网能正常访问
  54. 'openid' => $this->openid, // 用户id
  55. // 'out_trade_no' => '2015450806125348', // 商户订单号
  56. 'out_trade_no'=> $this->out_trade_no,
  57. //'spbill_create_ip' => $_SERVER['REMOTE_ADDR'], // 终端IP
  58. 'spbill_create_ip' => $this->spbill_create_ip, // 终端IP
  59. 'total_fee' => floatval(($this->total_fee) * 100), // 单位 分
  60. //'total_fee' => $this->total_fee, // 单位 分
  61. 'trade_type' => 'JSAPI' // 交易类型
  62. );
  63. //统一下单签名
  64. $parameters['sign'] = $this->getSign($parameters);
  65. $xmlData = $this->arrayToXml($parameters);
  66. $return = $this->xmlToArray($this->postXmlCurl($xmlData, $url, 60));
  67. //$return = $this->postXmlCurl($xmlData, $url, 60);
  68. // print_r($return);
  69. // die;
  70. return $return;
  71. }
  72. // curl请求方法封装
  73. private static function postXmlCurl($xml, $url, $second = 30)
  74. {
  75. $ch = curl_init();
  76. //设置超时
  77. curl_setopt($ch, CURLOPT_TIMEOUT, $second);
  78. curl_setopt($ch, CURLOPT_URL, $url);
  79. curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);
  80. curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); //严格校验
  81. //设置header
  82. curl_setopt($ch, CURLOPT_HEADER, FALSE);
  83. //要求结果为字符串且输出到屏幕上
  84. curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
  85. //post提交方式
  86. curl_setopt($ch, CURLOPT_POST, TRUE);
  87. curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
  88. curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 20);
  89. curl_setopt($ch, CURLOPT_TIMEOUT, 40);
  90. set_time_limit(0);
  91. //运行curl
  92. $data = curl_exec($ch);
  93. //返回结果
  94. if ($data) {
  95. curl_close($ch);
  96. return $data;
  97. } else {
  98. $error = curl_errno($ch);
  99. curl_close($ch);
  100. throw new WxPayException("curl出错,错误码:$error");
  101. }
  102. }
  103. //数组转换成xml
  104. private function arrayToXml($arr) {
  105. $xml = "<xml>";
  106. foreach ($arr as $key => $val) {
  107. if (is_array($val)) {
  108. $xml .= "<" . $key . ">" . arrayToXml($val) . "</" . $key . ">";
  109. } else {
  110. $xml .= "<" . $key . ">" . $val . "</" . $key . ">";
  111. }
  112. }
  113. $xml .= "</xml>";
  114. return $xml;
  115. }
  116. //xml转换成数组
  117. private function xmlToArray($xml) {
  118. //禁止引用外部xml实体
  119. libxml_disable_entity_loader(true);
  120. $xmlstring = simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA);
  121. $val = json_decode(json_encode($xmlstring), true);
  122. return $val;
  123. }
  124. //微信小程序接口
  125. private function weixinapp() {
  126. //统一下单接口
  127. $unifiedorder = $this->unifiedorder();
  128. // 统一下单出错,参数出错等原因
  129. if($unifiedorder['return_code'] == 'FAIL') {
  130. $retrunInfo['code'] = 0;
  131. $retrunInfo['msg'] = $unifiedorder['return_msg'];
  132. return $retrunInfo;
  133. }
  134. // print_r($unifiedorder);
  135. // die;
  136. $parameters = array(
  137. 'appId' => $this->appid, // 小程序ID
  138. 'timeStamp' => '' . time() . '', // 时间戳
  139. 'nonceStr' => $this->createNoncestr(), // 随机串
  140. 'package' => 'prepay_id=' . $unifiedorder['prepay_id'], // 数据包
  141. 'signType' => 'MD5' // 签名方式
  142. );
  143. // 小程序发起支付签名
  144. $parameters['paySign'] = $this->getSign($parameters);
  145. // 成功返回
  146. $retrunInfo['code'] = 1;
  147. $retrunInfo['msg'] = $parameters;
  148. return $retrunInfo;
  149. }
  150. //作用:产生随机字符串,不长于32位
  151. private function createNoncestr($length = 32) {
  152. $chars = "abcdefghijklmnopqrstuvwxyz0123456789";
  153. $str = "";
  154. for ($i = 0; $i < $length; $i++) {
  155. $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
  156. }
  157. return $str;
  158. }
  159. //作用:生成签名
  160. private function getSign($Obj) {
  161. foreach ($Obj as $k => $v) {
  162. $Parameters[$k] = $v;
  163. }
  164. //签名步骤一:按字典序排序参数
  165. ksort($Parameters);
  166. $String = $this->formatBizQueryParaMap($Parameters, false);
  167. //签名步骤二:在string后加入KEY
  168. $String = $String . "&key=" . $this->key;
  169. //签名步骤三:MD5加密
  170. $String = md5($String);
  171. //签名步骤四:所有字符转为大写
  172. $result_ = strtoupper($String);
  173. return $result_;
  174. }
  175. // 作用:格式化参数,签名过程需要使用
  176. private function formatBizQueryParaMap($paraMap, $urlencode) {
  177. $buff = "";
  178. ksort($paraMap);
  179. foreach ($paraMap as $k => $v) {
  180. if ($urlencode) {
  181. $v = urlencode($v);
  182. }
  183. $buff .= $k . "=" . $v . "&";
  184. }
  185. $reqPar;
  186. if (strlen($buff) > 0) {
  187. $reqPar = substr($buff, 0, strlen($buff) - 1);
  188. }
  189. return $reqPar;
  190. }
  191. }

四、发起请求接口的业务代码

  1. /**
  2. * **
  3. * author: fuchao
  4. * date: 2018-04-30
  5. * desc: 这里开始统一下单支付
  6. */
  7. $wxxcx_config = config('pay.wxxcx'); // 微信小程序设置
  8. $appid = $wxxcx_config['app_id']; // 小程序id
  9. $mch_id = $wxxcx_config['mch_id']; // 支付商户id
  10. $key = $wxxcx_config['md5_key']; // 商户的支付密钥
  11. $notify_url = $wxxcx_config['notify_url']; // 微信服务器异步通知
  12. $spbill_create_ip = $_SERVER['REMOTE_ADDR']; // 客户端ip
  13. $openid = $Xcxopenid; // 用户openid
  14. $out_trade_no = $Orderno; // 订单编号
  15. $body = $params['body']; // 订单描述
  16. $total_fee = $Alltotal; // 支付金额
  17. // var_dump($total_fee);
  18. // die;
  19. // 实例微信支付基类
  20. $weixinPay = new WeixinPay($appid, $openid, $mch_id, $key,$out_trade_no,$body,$total_fee,$notify_url,$spbill_create_ip);
  21. // 发起微信支付
  22. $result = $weixinPay->pay();
  23. if($result['code'] == 0) { // 统一下单出错
  24. return $this->sendError(1, $result['msg'], 200);
  25. }
  26. // 获取预支付返回参成功
  27. return $this->sendSuccess($result, 'success', 200);
  28. die;

五、异步通知(根据自己的业务逻辑) 

  1. /*微信支付的 异步通知 *回调地址*/
  2. /**回调修改2018-05-04**/
  3. public function notifyUrlApi() {
  4. //$xml = post_data();
  5. $xml = file_get_contents('php://input', 'r');
  6. //将服务器返回的XML数据转化为数组
  7. $data = $this->toArray($xml);
  8. // 判断签名是否正确 判断支付状态
  9. if (($data['return_code'] == 'SUCCESS') && ($data['result_code'] == 'SUCCESS')) {
  10. $result = $data;
  11. //获取服务器返回的数据
  12. $order_sn = $data['out_trade_no']; // 订单单号
  13. $openid = $data['openid']; // 付款人openID
  14. $total_fee = ($data['total_fee'])/100; // 付款金额
  15. $transaction_id = $data['transaction_id']; // 微信支付流水号
  16. //查找订单
  17. $order = Db::name('order')
  18. ->field('userid,status,order_type')
  19. ->where('status', 0) // 订单状态 0未支付 1支付成功 2取消订单
  20. ->where('order_no', $order_sn)
  21. ->find();
  22. if($order) { // 订单是否存在
  23. Db::startTrans();
  24. try {
  25. Db::name('order') // 更新订单状态(order)
  26. ->where('order_no', $order_sn)
  27. ->update(['transaction_no' => $transaction_id, 'status' => 1]);
  28. if ($order['order_type'] == 0) { // 更新圈子总金额
  29. $order_recharge_record = Db::name('order_recharge_record')
  30. ->where('order_no', $order_sn)
  31. ->find();
  32. Db::name('circle')
  33. ->where('id', $order_recharge_record['circleid'])
  34. ->setInc('total_amount', $total_fee);
  35. } else if ($order['order_type'] == 1) { // 更新用户金额
  36. Db::name('user')
  37. ->where('id', $order['userid'])
  38. ->setInc('balance', $total_fee);
  39. } else if ($order['order_type'] == 2) { // 更新任务状态
  40. $order_recharge_record = Db::name('order_recharge_record')
  41. ->where('order_no', $order_sn)
  42. ->find();
  43. $task_ok_UPDATE['ok'] = 1;
  44. $task_ok_UPDATE['ok_time'] = time();
  45. // 更新任务表
  46. Db::name('task')
  47. ->where('task_no', $order_recharge_record['taskno'])
  48. ->update($task_ok_UPDATE);
  49. // 更新任务详细记录表
  50. Db::name('task_record')
  51. ->where('task_no', $order_recharge_record['taskno'])
  52. ->update($task_ok_UPDATE);
  53. }else if ($order['order_type'] == 3) { // 更新vip状态
  54. $order_recharge_record = Db::name('order_recharge_record')
  55. ->where('order_no', $order_sn)
  56. ->find();
  57. $task_ok_UPDATE['ok'] = 1;
  58. $task_ok_UPDATE['ok_time'] = time();
  59. // 更新任务表
  60. Db::name('user_vip')
  61. ->where('vip_no', $order_recharge_record['vip_no'])
  62. ->update($task_ok_UPDATE);
  63. // 更新任务详细记录表
  64. Db::name('user_vip_record')
  65. ->where('vip_no', $order_recharge_record['vip_no'])
  66. ->update($task_ok_UPDATE);
  67. $Vipuserid = Db::name('user_vip_record')->field(true)->where('vip_no', $order_recharge_record['vip_no'])->select();
  68. $user_WHERE['id'] = ['in', array_column($Vipuserid, 'userid')];
  69. Db::name('user')->where($user_WHERE)->update(['vip' => 1]);
  70. }else if ($order['order_type'] == 4) { // 更新红包状态
  71. $order_recharge_record = Db::name('order_recharge_record')
  72. ->where('order_no', $order_sn)
  73. ->find();
  74. $task_ok_UPDATE['ok'] = 1;
  75. $task_ok_UPDATE['ok_time'] = time();
  76. // 更新任务表
  77. Db::name('redpacket')
  78. ->where('red_id', $order_recharge_record['red_id'])
  79. ->update($task_ok_UPDATE);
  80. }
  81. Db::commit();
  82. }catch (Exception $e) {
  83. $result = false;
  84. Db::rollback();
  85. }
  86. //$update['total_fee'] = $total_fee; // 保存支付成功的金额
  87. // $update['transaction_no'] = $transaction_id; // 保存支付商户号对应的ID号
  88. // $update['status'] = 1; // 订单状态 0未支付 1支付成功 2取消订单
  89. // /**更新订单**/
  90. // Db::name('order')
  91. // ->where('status', 0) // 订单状态 0未支付 1支付成功 2取消订单
  92. // ->where('order_no', $order_sn)
  93. // ->update($update);
  94. }else{ // 订单不存在
  95. $result = false;
  96. }
  97. }else {
  98. $result = false;
  99. }
  100. // 返回状态给微信服务器
  101. if ($result) {
  102. $str='<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>';
  103. }else{
  104. $str='<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[签名失败]]></return_msg></xml>';
  105. }
  106. echo $str;
  107. return $result;
  108. }

六、其他辅助方法(xml转数组

  1. /**
  2. * 将xml转为array
  3. * @param string $xml xml字符串
  4. * @return array 转换得到的数组
  5. */
  6. public function toArray($xml) {
  7. //禁止引用外部xml实体
  8. libxml_disable_entity_loader(true);
  9. $result= json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
  10. return $result;
  11. }

个人公众号

php对接微信小程序支付的更多相关文章

  1. 微信小程序支付步骤

    http://blog.csdn.net/wangsf789/article/details/53419781 最近开发微信小程序进入到支付阶段,一直以来从事App开发,所以支付流程还是熟记于心的.但 ...

  2. 微信小程序支付及退款流程详解

    微信小程序的支付和退款流程 近期在做微信小程序时,涉及到了小程序的支付和退款流程,所以也大概的将这方面的东西看了一个遍,就在这篇博客里总结一下. 首先说明一下,微信小程序支付的主要逻辑集中在后端,前端 ...

  3. Java 后端微信小程序支付demo (网上说的坑里面基本上都有)

    Java 后端微信小程序支付 一.遇到的问题 1. 商户号该产品权限未开通,请前往商户平台>产品中心检查后重试 2.签名错误 3.已经调起微信统一下单接口,可以拿到预支付ID,但是前端支付的时候 ...

  4. 微信小程序支付遇到的坑

    1,微信公众号支付和微信小程序支付有差异 微信公众号:可以直接跳转走h5的微信支付 微信小程序:在测试环境.沙箱环境使用微信公众号的跳转支付没有问题,在线上存在支付异常 最后商讨的解决方法 openi ...

  5. 微信小程序支付接入注意点

    一.微信支付后台服务器部署 服务器采用ubuntu16.04 + php7.0 + apache2.0. 微信支付后台服务使用了curl 和 samplexml ,因此php.ini配置中必须开启这两 ...

  6. 微信小程序支付开发之申请退款

    微信小程序支付跟微信公众号支付类似,这里不另做记录,如果没有开发过支付,可以查看我关于微信支付的文章 重点记录微信小程序申请退款开发过程中遇到一些坑. 退款接口比支付接口接口多了一个 双向证书 证书介 ...

  7. 微信小程序支付接入实战

    1. 微信小程序支付接入实战 1.1. 需求   最近接到一个小程序微信支付的需求,需要我写后台支持,本着能不自己写就不自己写的cv原则,在网上找到了些第三方程序,经过尝试后,最后决定了这不要脸作者的 ...

  8. 微信小程序支付异常:requestPayment:fail no permission

    今天在调试微信小程序支付时碰到了这个问题,支付参数都正常生成了,在调用 wx.requestPayment 进行支付时遇到了这个报错,查了一下发现是开发者工具中 AppID 写错了,用的 AppID ...

  9. SpringBoot2.0微信小程序支付多次回调问题

    SpringBoot2.0微信小程序支付多次回调问题 WxJava - 微信开发 Java SDK(开发工具包); 支持包括微信支付.开放平台.公众号.企业微信/企业号.小程序等微信功能的后端开发. ...

随机推荐

  1. 基于Oracle ADF的应用程序开发

    ADF简介 ADF(Application Development Framework)是Oracle公司为简化J2EE程序开发的复杂性专门开发的一种解决方案,ADF通过减少实现设计模式和应用程序框架 ...

  2. 数据挖掘进阶之序列模式挖掘GSP算法

    数据挖掘进阶之序列模式挖掘GSP算法 绪 继续数据挖掘方面算法的讲解,前面讲解了数据挖掘中关联规则算法FP-Growth的实现.此篇博文主要讲解基于有趣性度量标准的GSP序列模式挖掘算法.有关论文后期 ...

  3. Mybatis的resultType

    使用mybatis去查询数据时,没有指定resultType,mybatis无法返回正常结果,当然在web中并没有出现报错,所以有点坑自己了,所以需要使用如下配置: <select id=&qu ...

  4. Redis服务信息

    想要获得下面的redis服务器信息,只需要在命令行中输入:info server 部分记录了 Redis 服务器的信息,它包含以下域: redis_version : Redis 服务器版本 redi ...

  5. NOSQL schema创建原则

    (1)数据规模 Bigtable类数据库系统(HBase,Cassandra等)是为了解决海量数据规模的存储需要设计的.这里说的海量数据规模指的是单个表存储的数据量是在TB或者PB规模,单个表是由千亿 ...

  6. Android逆向分析(2) APK的打包与安装背后的故事

    前言 上一次我们反编译了手Q,并遇到了Apktool反编译直接crash的问题,虽然笔者很想在这次解决这个问题,但在解决途中,发现该保护依赖于很多知识,所以本次先插入一下,正所谓知其然知其所以然,授之 ...

  7. rails将类常量重构到数据库对应的表中之三

    经过博文之一和之二的重构,貌似代码表现的还不错,正常运行和test都通过鸟,但是,感觉告诉我们还是有什么地方不对劲啊!究竟是哪里不对劲呢?我们再来好好看一下. 我们把数据库表中的支付方式集合直接放在实 ...

  8. x&(x-1)

    x&(x-1)可以用来求出x是否为2幂次方数:当&的结果为0时,x原值是2幂次方数,否则就不是2幂次方数: x=x&(x-1)即把x从低位开始的第一个1改成0.如1000,把1 ...

  9. 直接执行SQL语句的快捷键是什么啊?嘎嘎

    在查询中输入SQL语句后,执行语句的快捷键~ 分享到: 2009-10-23 10:59网友采纳 左键..嘿嘿,开个玩笑 你把鼠标移动到执行按钮上停一会就能看到了啊

  10. PhpStorm服务激活

    日期 服务地址 状态  2018-03-15  http://idea.singee77.com/  使用中