本文实例为大家分享了php实现小程序支付的具体代码,供大家参考,具体内容如下

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

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));
}
/**
* 支付回调
* @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>
 

PHP实现微信小程序支付完整版,可以借鉴!的更多相关文章

  1. Java实现微信小程序支付(完整版)

    在开发微信小程序支付的功能前,我们先熟悉下微信小程序支付的业务流程图: 不熟悉流程的建议还是仔细阅读微信官方的开发者文档. 一,准备工作 事先需要申请企业版小程序,并开通“微信支付”(即商户功能).并 ...

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

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

  3. Asp.net Core 微信小程序支付

    最近要做一个微信小程序支付的功能 在网上找了一下 .net Core做微信支付的博客 和 demo 几乎没有 自己研究了好几天 参考了 很多 大牛的博客 勉强做出来了  因为参数都没有 比如 opid ...

  4. 微信小程序支付(企业支付给用户零钱)

    内容摘要:本案例客户端支付方式为微信小程序支付(JSAPI).商户运营一段时间后,在微信商户平台开通企业支付服务后,即可调用微信支付提供的企业付款接口将佣金等金额通过微信零钱返现给C端用户零钱. 服务 ...

  5. 微信小程序支付步骤

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

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

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

  7. php对接微信小程序支付

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

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

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

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

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

随机推荐

  1. [JZOJ3296] 【SDOI2013】刺客信条

    题目 题目大意 给你一棵树,树上每个节点有000或111的状态. 用最少的操作次数使得当前状态与目标状态同构. 思考历程 首先想到的是找重心. 因为根是不确定的,但重心只会有一个或两个,以重心为根就能 ...

  2. SaaS加速器,到底加速了谁? 剖析阿里云的SaaS战略:企业和ISV不可错过的好文

    过去二十年,中国诞生了大批To C的高市值互联网巨头,2C的领域高速发展,而2B领域一直不温不火.近两年来,在C端流量饱和,B端数字化转型来临的背景下,中国越来越多的科技公司已经慢慢将触角延伸到了B端 ...

  3. Ubuntu GitHub操作——分支、合并与标签

    分支 分支是用来将特性开发绝缘开来的.在你创建仓库的时候,master 是"默认的"分支.在其他分支上进行开发,完成后再将它们合并到主分支上. 创建一个叫做"featur ...

  4. dijkstra (模板)

    突然意识到以前写的都是假的dij,感谢GhostCai神犇. #include<iostream> #include<cstdio> #include<cstring&g ...

  5. Python学习day36-并发编程(2)

    figure:last-child { margin-bottom: 0.5rem; } #write ol, #write ul { position: relative; } img { max- ...

  6. 启动zuul时候报错:The bean 'proxyRequestHelper', defined in class path resource [org/springframework/cloud/netflix/zuul

    启动zuul时候报错:The bean 'proxyRequestHelper', defined in class path resource [org/springframework/cloud/ ...

  7. [转]【全面解禁!真正的Expression Blend实战开发技巧】第八章 FluidMoveBehavior完全解析之一漂浮移动

    好久没更新博客了,今天如果没急事,准备连发三篇,完全讲解Blend最牛的元素-“FluidMoveBehavior”.我向大家保证这三章一定非常精彩,不看你肯定后悔.我相信这三篇文章发表后,国内很多s ...

  8. linux拆分文件

    1.先看下文件总的行数: wc -l filename 我们现在来看看它具体的参数该怎么用: split支持自定义输出文件大小和输出文件行数两种模式,此外还可以定义每一行最大的值. -l 按输出文件行 ...

  9. mapreduce.Job: Running job: job_1553100392548_0001

    这几天一直在尝试一个mapreduce的例子,但是一直都是卡在mapreduce.Job: Running job: job_1553100392548_0001,查看日志也不报错,查看每个配置文件没 ...

  10. 子类A继承父类B, A a = new A(); 则父类B构造函数、父类B静态代码块、父类B非静态代码块、子类A构造函数、子类A静态代码块、子类A非静态代码块 执行的先后顺序是

    按照先后顺序: 1,静态先于非静态代码库执行(静态代码块随着类的加载而加载,初始化只执行一次) 2,父类先于子类 3,非静态代码块优于构造函数执行 所以执行顺序如下: 父类B静态代码块->子类A ...