PHP小程序支付功能完整版【基于thinkPHP】


Posted in PHP onMarch 26, 2019

本文实例讲述了PHP小程序支付功能。分享给大家供大家参考,具体如下:

环境: tp3.2  + 小程序 微信支付功能开通

Step1:  下载PHP 支付SDK(下载地址)  放到Library\Vendor下,取名Wxpay

修改WxPay.Config.php 里的appid appsecret key MCHID

Step2: 小程序 js 代码:

var url = getApp().globalData.httpServer + 'api/buy/pay';
var userId = getApp().globalData.userId;
var totalMoney = this.data.totalMoney;
var cart = this.data.goods;
var param = {
  cart: JSON.stringify(cart),
  cartamount: totalMoney,
  userid: userId,
  payment: this.data.payment,
  addressid: defaultAddress.id
};
var that = this;
util.http(url, param, function (ret) {
  if (ret.data.code == 1) {
    if (that.data.payment == 'balance') { // 余额支付
      that.afterPaySuccess(ret.data.data);
    } else {                // 微信支付
      wx.requestPayment({
        timeStamp: ret.data.data.timeStamp,
        nonceStr: ret.data.data.nonceStr,
        package: ret.data.data.package,
        signType: ret.data.data.signType,
        paySign: ret.data.data.paySign,
        'success': function (res) {
          that.afterPaySuccess(ret.data.data.orderid);
        },
        'fail': function (res) {
          console.log(res);
        }
      })
    }
  } else {
    util.showTip(ret.data.msg, '提交订单失败');
  }
});
/**
 * 网络请求
 */
function http(url, params, callback) {
  wx.request({
    url: url,
    data: params,
    success: function (res) {
      callback(res);
    },
    fail: function (err) {
      console.log(err);
    }
  });
}

Step3: 接口代码:

public function pay()
{
    $cart = I('cart', '', 'trim');
    $cartAmount = I('cartamount');
    $addressId = I('addressid', 0, 'intval');
    $payment = I('payment', '', 'trim');
    $userId = $this->userid;
    $cart = json_decode($cart, true);
    if (empty($cart)) {
      $result['msg'] = '购物车获取失败';
      $result['code'] = 0;
      $this->ajaxReturn($result);
    }
    $totalMoney = 0;
    foreach ($cart as $goods) {
      $money = $goods['price'];  // price
      $selectCount = $goods['selectcount'];  // price
      $itemAmount = number_format($money * $selectCount, 2, '.', '');
      $totalMoney += $itemAmount;
    }
    // 检查总金额是否一致
    if ($totalMoney != $cartAmount) {
      $result['msg'] = '总金额不匹配:' . $totalMoney;
      $result['code'] = 0;
      $this->ajaxReturn($result);
    }
    // 获取用户地址
    $address = M('MemberAddress')->where('userid=' . $userId . " and id=" . $addressId)->find();
    if (empty($address)) {
      $result['msg'] = '用户地址不存在';
      $result['code'] = 0;
      $this->ajaxReturn($result);
    }
    // 用户信息
    $user = M('Member')->where("id=" . $userId)->find();
    if ($payment == 'balance') {
      if ($user['amount'] < $cartAmount) {
        $result['msg'] = '余额不足';
        $result['code'] = 0;
        $this->ajaxReturn($result);
      }
    }
    // 生成订单
    $order['ordersn'] = $this->genOrdersn($user['id']);
    $order['price'] = $cartAmount;
    $order['addressid'] = $address['id'];
    $order['addressinfo'] = serialize($address); //json_encode($address);
    $order['longitude'] = $address['longitude'];
    $order['latitude'] = $address['latitude'];
    $order['addtime'] = time();
    $order['status'] = 0;
    $order['userid'] = $user['id'];
    $order['paytype'] = $payment;
    $order['paysn'] = '';
    $order['paytime'] = time();
    $orderId = M("Order")->add($order);
    if ($orderId == 0) {
      $result['msg'] = '创建订单失败';
      $result['code'] = 0;
      $this->ajaxReturn($result);
    }
    foreach ($cart as $goods) {
      $orderGoods['orderid'] = $orderId;
      $orderGoods['goodsid'] = $goods['id'];
      $orderGoods['title'] = $goods['title'];
      $orderGoods['price'] = $goods['price'];
      $orderGoods['attr'] = $goods['attr'];
      $orderGoods['pic'] = $goods['pic'];
      $orderGoods['num'] = $goods['selectcount'];
      M("OrderGoods")->add($orderGoods);
    }
    if ($payment == 'balance') {
      // 余额支付
      $this->balancePay($cartAmount, $user['wxopenid'], $orderId);
    } else if ($payment == 'weixin') {
      // 微信支付
      $this->weixinPay($cartAmount, $user['wxopenid'], $orderId, $order['ordersn']);
    }
}
/**
* 微信支付
* @author 大脸猫脸大
* @param $cart
* @param $cartAmount
* @param $address
* @param $user
*/
private function weixinPay($cartAmount, $openId, $orderId, $orderSn)
{
    import("Vendor.Wxpay.lib.WxPay#Api", "", ".php");
    //订单号
    $money = $cartAmount * 100;
    $openid = $openId;
    $input = new \WxPayUnifiedOrder();
    $input->SetBody("迪克-商品");
    $input->SetOut_trade_no("$orderSn");
    $input->SetTotal_fee("$money");
    $input->SetNotify_url("https://" . $_SERVER['HTTP_HOST'] . "/api/buy/payNotify");
    $input->SetTrade_type("JSAPI");
    $input->SetOpenid($openid);
    $unifiedOrder = \WxPayApi::unifiedOrder($input);
    if ($unifiedOrder['result_code'] == 'SUCCESS' && $unifiedOrder['return_code'] == 'SUCCESS') {
      $time = time();
      $data['timeStamp'] = "$time";              //时间戳
      $data['nonceStr'] = $unifiedOrder['nonce_str'];     //随机字符串
      $data['signType'] = 'MD5';                //签名算法,暂支持 MD5
      $data['package'] = 'prepay_id=' . $unifiedOrder['prepay_id'];  //统一下单接口返回的 prepay_id 参数值,提交格式如:prepay_id=*
      $data['paySign'] = $this->genPaySign($unifiedOrder, $time);// 之前以为是$unifiedOrder['sign']; 后来发现是调用的这种方法. 签名方案参见微信公众号支付帮助文档;
      $data['out_trade_no'] = $orderSn;
      $data['orderid'] = $orderId;
      $return['code'] = 1;
      $return['data'] = $data;
    } else {
        Log::write(var_export($unifiedOrder, true), Log::ERR, '', C('LOG_PATH')."wx_pay_".date('y_m_d').'.log');
        $return['code'] = 0;
        $return['msg'] = '微信支付失败';// $unifiedOrder['RETURN_MSG'];
    }
    $this->ajaxReturn($return);
}
/* 生成支付签名*/
private function genPaySign($unifiedOrder, $time)
{
    $appId = \WxPayConfig::APPID;
    $nonceStr = $unifiedOrder['nonce_str'];
    $package = 'prepay_id=' . $unifiedOrder['prepay_id'];
    $signType = "MD5";
    $timeStamp = $time;
    $key = \WxPayConfig::KEY;
    $sign = md5(sprintf("appId=%s&nonceStr=%s&package=%s&signType=%s&timeStamp=%s&key=%s", $appId, $nonceStr, $package, $signType, $timeStamp, $key));
    return $sign;
}
/**
* 支付回调
* @author:大脸猫脸大
*/
public function payNotify()
{
    import("Vendor.Wxpay.lib.WxPay#Data", "", ".php");
    $xml = $GLOBALS['HTTP_RAW_POST_DATA'];
    $val = \WxPayResults::Init($xml);
    if ($val['result_code'] == 'SUCCESS' && $val['return_code'] == 'SUCCESS') {
      $orderSn = $val['out_trade_no'];
      $transactionId = $val['transaction_id'];
      $data = array('paytype' => 'weixin', 'status' => '1', 'paytime' => time(), 'paysn' => $transactionId);
      M("Order")->where("ordersn='$orderSn'")->setField($data);
      exit('<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>');
    }else {
      Log::write(var_export($val, true), Log::ERR, '', C('LOG_PATH')."wx_pay_notify_".date('y_m_d').'.log');
      exit('<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>');
    }
}

payNotify 回调方法里一定要注意返回

<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>

如果不处理,你会发现payNotify  会被执行很多次参见:官方文档

总结一下: 注意二点, 1.签名的问题 2. 回调方法的返回处理。

欢迎大家指正。

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
PHPShop存在多个安全漏洞
Oct 09 PHP
如何在PHP中使用Oracle数据库(6)
Oct 09 PHP
深入for,while,foreach遍历时间比较的详解
Jun 08 PHP
PHP cdata 处理(详细介绍)
Jul 05 PHP
PHP5中实现多态的两种方法实例分享
Apr 21 PHP
CodeIgniter中实现泛域名解析
Jul 19 PHP
PHP查询并删除数据库多列重复数据的方法(利用数组函数实现)
Feb 23 PHP
Thinkphp框架开发移动端接口(1)
Aug 18 PHP
WAF的正确bypass
Jan 05 PHP
PHP7.0连接DB操作实例分析【基于mysqli】
Sep 26 PHP
php使用event扩展的io复用测试的示例
Oct 20 PHP
详解Go与PHP的语法对比
May 29 PHP
PHP实现微信退款的方法示例
Mar 26 #PHP
PHP设计模式之单例模式定义与用法分析
Mar 26 #PHP
php+mysql开发中的经验与常识小结
Mar 25 #PHP
PHP设计模式之抽象工厂模式实例分析
Mar 25 #PHP
PHP设计模式之简单工厂和工厂模式实例分析
Mar 25 #PHP
PHP实现无限极分类的两种方式示例【递归和引用方式】
Mar 25 #PHP
详解PHP神奇又有用的Trait
Mar 25 #PHP
You might like
玛琪朵 Macchiato
2021/03/03 咖啡文化
使用PHP数组实现无限分类,不使用数据库,不使用递归.
2006/12/09 PHP
域名和cookie问题(域名后缀)
2012/10/10 PHP
PHP ajax 异步执行不等待执行结果的处理方法
2015/05/27 PHP
PHP使用stream_context_create()模拟POST/GET请求的方法
2016/04/02 PHP
php中序列化与反序列化详解
2017/02/13 PHP
php实现等比例压缩图片
2018/07/26 PHP
YII框架行为behaviors用法示例
2019/04/26 PHP
Avengerls vs KG BO3 第二场2.18
2021/03/10 DOTA
jquery获得下拉框值的代码
2011/08/13 Javascript
JavaScript利用构造函数和原型的方式模拟C#类的功能
2014/03/06 Javascript
JavaScript将取代AppleScript?
2014/09/18 Javascript
使用jQuery实现图片遮罩半透明坠落遮挡
2015/03/16 Javascript
javascript实现百度地图鼠标滑动事件显示、隐藏
2015/04/02 Javascript
jQuery实现带有洗牌效果的动画分页实例
2015/08/31 Javascript
JavaScript获取当前运行脚本文件所在目录的方法
2016/02/03 Javascript
js判断鼠标位置是否在某个div中的方法
2016/02/26 Javascript
jquery对象访问是什么及使用方法介绍
2016/05/03 Javascript
Bootstrap3 datetimepicker控件使用实例
2016/12/13 Javascript
快速实现JS图片懒加载(可视区域加载)示例代码
2017/01/04 Javascript
javascript 中iframe高度自适应(同域)实例详解
2017/05/16 Javascript
详解Vue组件之间的数据通信实例
2017/06/17 Javascript
原生js FileReader对象实现图片上传本地预览效果
2020/03/27 Javascript
Puppeteer 爬取动态生成的网页实战
2018/11/14 Javascript
微信端调取相册和摄像头功能,实现图片上传,并上传到服务器
2019/05/16 Javascript
Node.js系列之连接DB的方法(3)
2019/08/30 Javascript
[36:52]DOTA2真视界:基辅特锦赛总决赛
2017/05/21 DOTA
Python数据持久化shelve模块用法分析
2018/06/29 Python
python使用matplotlib画饼状图
2018/09/25 Python
Python MySQLdb 执行sql语句时的参数传递方式
2020/03/04 Python
详解python百行有效代码实现汉诺塔小游戏(简约版)
2020/10/30 Python
Python绘制K线图之可视化神器pyecharts的使用
2021/03/02 Python
英国最大的在线运动补充剂商店:Discount Supplements
2017/06/03 全球购物
爱尔兰领先的在线体育用品零售商:theGAAstore
2018/04/16 全球购物
新品发布会策划方案
2014/06/08 职场文书
python源码剖析之PyObject详解
2021/05/18 Python