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 相关文章推荐
php4的session功能评述(二)
Oct 09 PHP
PHP实现用户认证及管理完全源码
Mar 11 PHP
php对数组排序代码分享
Feb 24 PHP
CI框架学习笔记(二) -入口文件index.php
Oct 27 PHP
一款简单实用的php操作mysql数据库类
Dec 08 PHP
[原创]ThinkPHP让../Public在模板不解析(直接输出)的方法
Oct 09 PHP
Symfony2实现从数据库获取数据的方法小结
Mar 18 PHP
Zend Framework教程之Zend_Db_Table_Row用法实例分析
Mar 21 PHP
CI框架文件上传类及图像处理类用法分析
May 18 PHP
PHP实现的登录,注册及密码修改功能分析
Nov 25 PHP
PHP实现给定一列字符,生成指定长度的所有可能组合示例
Jun 22 PHP
laravel解决迁移文件一次删除创建字段报错的问题
Oct 24 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
PHP访问MYSQL数据库封装类(附函数说明)
2010/12/04 PHP
php rsa 加密,解密,签名,验签详解
2016/12/06 PHP
laravel框架 laravel-admin上传图片到oss的方法
2019/10/13 PHP
不用AJAX和IFRAME,说说真正意义上的ASP+JS无刷新技术
2008/09/25 Javascript
LazyForm jQuery plugin 定制您的CheckBox Radio和Select
2009/10/24 Javascript
JQuery从头学起第二讲
2010/07/04 Javascript
De facto standard 世界上不可思议的事实标准
2010/08/29 Javascript
JS判断不同分辨率调用不同的CSS样式文件实现思路及测试代码
2013/01/23 Javascript
jquery实现智能感知连接外网搜索
2013/05/21 Javascript
JS 按钮点击触发(兼容IE、火狐)
2013/08/07 Javascript
使用console进行性能测试
2015/04/27 Javascript
JQuery给网页更换皮肤的方法
2015/05/30 Javascript
使用ngView配合AngularJS应用实现动画效果的方法
2015/06/19 Javascript
js编写贪吃蛇的小游戏
2020/08/24 Javascript
Django静态资源URL STATIC_ROOT的配置方法
2014/11/08 Python
Python实现登录接口的示例代码
2017/07/21 Python
Python格式化输出%s和%d
2018/05/07 Python
python 监听salt job状态,并任务数据推送到redis中的方法
2019/01/14 Python
解决使用Pandas 读取超过65536行的Excel文件问题
2020/11/10 Python
利用CSS3实现平移动画效果示例代码
2016/10/12 HTML / CSS
HTML5 离线应用之打造零请求、无流量网站的解决方法
2013/04/25 HTML / CSS
GOOD AMERICAN官网:为曲线性感而设计
2017/12/28 全球购物
美国奢侈品购物平台:Orchard Mile
2018/05/02 全球购物
高一家长会邀请函
2014/01/12 职场文书
装修五一活动策划案
2014/01/23 职场文书
《沙漠中的绿洲》教学反思
2014/04/24 职场文书
优秀会计求职信
2014/07/04 职场文书
声乐专业大学生职业生涯规划书:理想的未来需要自己去打造
2014/09/20 职场文书
惹女朋友生气检讨书
2015/05/06 职场文书
迎新生欢迎词2015
2015/07/16 职场文书
大学生党员暑假实践(活动总结)
2019/08/21 职场文书
浅谈Python列表嵌套字典转化的问题
2021/04/07 Python
8个JS的reduce使用实例和reduce操作方式
2021/10/05 Javascript
Consul在linux环境的集群部署
2022/04/08 Servers
Spring Boot接口定义和全局异常统一处理
2022/04/20 Java/Android
Redis特殊数据类型bitmap位图
2022/06/01 Redis