微信小程序支付PHP代码


Posted in Javascript onAugust 23, 2018

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

服务器端获取 openid

Getopenid.php

<?php
 header('Content-type: application/json; charset=UTF-8');
 $APPID="";//填写小程序appid
 $SECRET="";//填写小程序secret
 $JSCODE="";
 if(isset($_GET['js_code'])){
  $JSCODE=$_GET['js_code'];
  $url="https://api.weixin.qq.com/sns/jscode2session?appid=".$APPID
  ."&secret=".$SECRET."&js_code=".$JSCODE."&grant_type=authorization_code";
  $curl = curl_init();
  curl_setopt($curl, CURLOPT_URL, $url);
  curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
  curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
  curl_setopt($curl, CURLOPT_HEADER, 0);
  $data = curl_exec($curl);
  $array=json_decode($data,true);

  curl_close($curl);
  $openid=isset($array['openid'])?$array['openid']:$array['errcode'];
  if($openid=="40029"){
   $response["result"] = 0;
   $response["msg"] = "invalid code";
   $response["openid"] = $openid;
   echo json_encode($response);
  }else{
   $response["result"] = 1;
   $response["msg"] = "user exist";
   $response["openid"] = $openid;
   echo json_encode($response);
  }
 }

小程序存储openid
在app.js中

getUserInfo:function(cb){
 var that = this
 if(this.globalData.userInfo){
  typeof cb == "function" && cb(this.globalData.userInfo)
 }else{
  wx.login({
  success: function (res) {
   if (res.code) {
   var code = res.code;
   wx.getUserInfo({
    success: function (res2) {
    console.log(res2);
    that.globalData.userInfo = res2.userInfo;
    typeof cb == "function" && cb(that.globalData.userInfo)
    var encryptedData = encodeURIComponent(res2.encryptedData);//一定要把加密串转成URI编码
    var iv = res2.iv;
    //请求自己的服务器
    //Login(code, encryptedData, iv);
    wx.showToast({
     title: '正在登录...',
     icon: 'loading',
     duration: 10000
    });
    //请求服务器
    wx.request({
     url: API_URL,//Getopenid.php
     data: {
     js_code: code,
     },
     method: 'GET',
     header: {
     'content-type': 'application/json'
     }, // 设置请求的 header
     success: function (res) {
     // success
     wx.hideToast();
     console.log("JSON:" + res.data);
     if (res.data.result=="1"){//获取openid成功
      wx.setStorage({//存储openid
      key: "openid",
      data: res.data.openid
      })
     }else{
      wx.showToast({
      title: 'openid获取失败',
      icon: 'none',
      duration: 2000
      })
     }
     console.log('服务器返回' + res.data.result);
     console.log('服务器返回' + res.data.msg);
     console.log('服务器返回' + res.data.openid);
     },
     fail: function () {
     // fail
     // wx.hideToast();
     },
     complete: function () {
     // complete
     }
    })
    }
   })
   } else {
   console.log('获取用户登录态失败!' + res.errMsg)
   }
  }
  })
 }
 }

在登陆界面获取openid

var app = getApp()
onLoad: function () {
 console.log('onLoad')
 var that = this
 //调用应用实例的方法获取全局数据
 app.getUserInfo(function(userInfo){//获取用户信息 
  //更新数据
  that.setData({
  userInfo:userInfo
  })
 })
}

通过以上步骤已经获取到openid

支付方法小程序

pay() {
 var that = this;
 if (this.data.totalPrice == 0) {
  return;
 }
 wx.getStorage({//获取存储在本地的openid
  key: 'openid',
  success: function (res) {
  console.log(res.data)
  that.setData({
   openid:res.data,
  })
  var carArray = that.data.carArray;
  var str="";
  for (var i = 0; i < carArray.length; i++) {
   str=str+ carArray[i].num+"个" + carArray[i].name+" ";
  }
  wx.request({
   url: 'Pay.php',//支付接口
   data: {
   openid: res.data,//openid
   total_fee: that.data.totalPrice,//总金额
   body: str,//商品描述
   },
   method:'GET',
   success:function(res){
   console.log(res.data['timeStamp'])
   if(res.data){
    wx.requestPayment({
    'timeStamp': res.data['timeStamp'],
    'nonceStr': res.data['nonceStr'],
    'package': res.data['package'],
    'signType': 'MD5',
    'paySign': res.data['paySign'],
    'success': function (res) {
     wx.showToast({
     title: '支付成功',
     icon: 'succes',
     duration: 1000,
     mask: true
     })
     //支付成功后在数据库减去购买商品的数量
     var carArray = that.data.carArray;
     for (var i = 0; i < carArray.length; i++) {
     that.setData({
      jiesuan_num: carArray[i].num,
      jiesuan_id: carArray[i].goods_id,
     })
     DeGood(that);
     }
    },
    'fail': function (res) {
     wx.showToast({
     title: '支付失败',
     icon: 'none',
     duration: 1000,
     mask: true
     })
    }
    })
   }
   }
  })
  }
 })
 },

Pay.php

<?php
 include 'WeixinPay.php'; 
 $appid=''; //小程序appid
 $openid= $_GET['openid']; 
 $mch_id=''; //商户id
 $key=''; //商户key
 $out_trade_no = $mch_id. time(); 
 $total_fee = $_GET['total_fee']; 
 $body= $_GET['body'];
 if(empty($total_fee)){ 
  $body = $body; 
  $total_fee = floatval(99*100); 
 }else{ 
  $body = $body; 
  $total_fee = floatval($total_fee*100); 
 } 
 $weixinpay = new WeixinPay($appid,$openid,$mch_id,$key,$out_trade_no,$body,$total_fee); 
 $return=$weixinpay->pay(); 
 echo json_encode($return);

WeixinPay.php

<?php


/*
 * 小程序微信支付
 */


class WeixinPay {


 protected $appid;
 protected $mch_id;
 protected $key;
 protected $openid;
 protected $out_trade_no;
 protected $body;
 protected $total_fee;
   function __construct($appid, $openid, $mch_id, $key,$out_trade_no,$body,$total_fee) {
  $this->appid = $appid;
  $this->openid = $openid;
  $this->mch_id = $mch_id;
  $this->key = $key;
  $this->out_trade_no = $out_trade_no;
  $this->body = $body;
  $this->total_fee = $total_fee;
 }


 public function pay() {
  //统一下单接口
  $return = $this->weixinapp();
  return $return;
 }


 //统一下单接口
 private function unifiedorder() {
  $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
  $parameters = array(
   'appid' => $this->appid, //小程序ID
   'mch_id' => $this->mch_id, //商户号
   'nonce_str' => $this->createNoncestr(), //随机字符串
//   'body' => 'test', //商品描述
   'body' => $this->body,
//   'out_trade_no' => '2015450806125348', //商户订单号
   'out_trade_no'=> $this->out_trade_no,
//   'total_fee' => floatval(0.01 * 100), //总金额 单位 分
   'total_fee' => $this->total_fee,
//   'spbill_create_ip' => $_SERVER['REMOTE_ADDR'], //终端IP
   'spbill_create_ip' => '192.168.0.161', //终端IP
   'notify_url' => 'http://www.weixin.qq.com/wxpay/pay.php', //通知地址 确保外网能正常访问
   'openid' => $this->openid, //用户id
   'trade_type' => 'JSAPI'//交易类型
  );
  //统一下单签名
  $parameters['sign'] = $this->getSign($parameters);
  $xmlData = $this->arrayToXml($parameters);
  $return = $this->xmlToArray($this->postXmlCurl($xmlData, $url, 60));
  return $return;
 }


 private static function postXmlCurl($xml, $url, $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);


  curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 20);
  curl_setopt($ch, CURLOPT_TIMEOUT, 40);
  set_time_limit(0);


  //运行curl
  $data = curl_exec($ch);
  //返回结果
  if ($data) {
   curl_close($ch);
   return $data;
  } else {
   $error = curl_errno($ch);
   curl_close($ch);
   throw new WxPayException("curl出错,错误码:$error");
  }
 }



 //数组转换成xml
 private function arrayToXml($arr) {
  $xml = "<root>";
  foreach ($arr as $key => $val) {
   if (is_array($val)) {
    $xml .= "<" . $key . ">" . arrayToXml($val) . "</" . $key . ">";
   } else {
    $xml .= "<" . $key . ">" . $val . "</" . $key . ">";
   }
  }
  $xml .= "</root>";
  return $xml;
 }


 //xml转换成数组
 private function xmlToArray($xml) {


  //禁止引用外部xml实体 


  libxml_disable_entity_loader(true);


  $xmlstring = simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA);


  $val = json_decode(json_encode($xmlstring), true);


  return $val;
 }


 //微信小程序接口
 private function weixinapp() {
  //统一下单接口
  $unifiedorder = $this->unifiedorder();
//  print_r($unifiedorder);
  $parameters = array(
   'appId' => $this->appid, //小程序ID
   'timeStamp' => '' . time() . '', //时间戳
   'nonceStr' => $this->createNoncestr(), //随机串
   'package' => 'prepay_id=' . $unifiedorder['prepay_id'], //数据包
   'signType' => 'MD5'//签名方式
  );
  //签名
  $parameters['paySign'] = $this->getSign($parameters);
  return $parameters;
 }


 //作用:产生随机字符串,不长于32位
 private function createNoncestr($length = 32) {
  $chars = "abcdefghijklmnopqrstuvwxyz0123456789";
  $str = "";
  for ($i = 0; $i < $length; $i++) {
   $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
  }
  return $str;
 }


 //作用:生成签名
 private function getSign($Obj) {
  foreach ($Obj as $k => $v) {
   $Parameters[$k] = $v;
  }
  //签名步骤一:按字典序排序参数
  ksort($Parameters);
  $String = $this->formatBizQueryParaMap($Parameters, false);
  //签名步骤二:在string后加入KEY
  $String = $String . "&key=" . $this->key;
  //签名步骤三:MD5加密
  $String = md5($String);
  //签名步骤四:所有字符转为大写
  $result_ = strtoupper($String);
  return $result_;
 }


 ///作用:格式化参数,签名过程需要使用
 private function formatBizQueryParaMap($paraMap, $urlencode) {
  $buff = "";
  ksort($paraMap);
  foreach ($paraMap as $k => $v) {
   if ($urlencode) {
    $v = urlencode($v);
   }
   $buff .= $k . "=" . $v . "&";
  }
  $reqPar;
  if (strlen($buff) > 0) {
   $reqPar = substr($buff, 0, strlen($buff) - 1);
  }
  return $reqPar;
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
js判断背景图片是否加载成功使用img的width实现
May 29 Javascript
jQuery中append()方法用法实例
Dec 25 Javascript
浅谈javascript 函数属性和方法
Jan 21 Javascript
js实现从中间开始往上下展开网页窗口的方法
Mar 02 Javascript
js window对象属性和方法相关资料整理
Nov 11 Javascript
JavaScript中的数组遍历forEach()与map()方法以及兼容写法介绍
May 19 Javascript
微信小程序 实现动态显示和隐藏某个控件
Apr 27 Javascript
Angualrjs 表单验证的两种方式(失去焦点验证和点击提交验证)
May 09 Javascript
VUE元素的隐藏和显示(v-show指令)
Jun 23 Javascript
layui-table表复选框勾选的所有行数据获取的例子
Sep 13 Javascript
jquery实现简单每周轮换的日历
Sep 10 jQuery
jquery实现穿梭框功能
Jan 19 jQuery
vue+axios 前端实现的常用拦截的代码示例
Aug 23 #Javascript
详解如何配置vue-cli3.0的vue.config.js
Aug 23 #Javascript
JS使用Date对象实时显示当前系统时间简单示例
Aug 23 #Javascript
React key值的作用和使用详解
Aug 23 #Javascript
vue项目中使用lib-flexible解决移动端适配的问题解决
Aug 23 #Javascript
从零开始学习搭建React脚手架项目
Aug 23 #Javascript
angular2 ng2-file-upload上传示例代码
Aug 23 #Javascript
You might like
咖啡知识 咖啡养豆要养多久 排气又是什么
2021/03/06 新手入门
PHP+javascript模拟Matrix画面
2006/10/09 PHP
laravel-admin表单提交隐藏一些数据,回调时获取数据的方法
2019/10/08 PHP
PHP实现文件上传与下载
2020/08/28 PHP
JavaScript继承方式实例
2010/10/29 Javascript
JavaScript替换当前页面的方法
2015/04/03 Javascript
JS+Canvas绘制时钟效果
2020/08/20 Javascript
Jquery针对tr td的一些实用操作方法(必看篇)
2016/10/05 Javascript
jQuery实现一个简单的轮播图
2017/02/19 Javascript
从零学习node.js之简易的网络爬虫(四)
2017/02/22 Javascript
微信小程序 转发功能的实现
2017/08/04 Javascript
详解JavaScript 的变量
2019/03/08 Javascript
如何根据业务封装自己的功能组件
2019/04/19 Javascript
全面分析JavaScript 继承
2019/05/30 Javascript
[53:38]OG vs LGD 2018国际邀请赛淘汰赛BO3 第三场 8.26
2018/08/30 DOTA
python动态参数用法实例分析
2015/05/25 Python
Python数据类型详解(二)列表
2016/05/08 Python
Python日期的加减等操作的示例
2017/08/15 Python
Python中static相关知识小结
2018/01/02 Python
python如何把嵌套列表转变成普通列表
2018/03/20 Python
儿童python练习实例
2018/05/27 Python
对Python2与Python3中__bool__方法的差异详解
2018/11/01 Python
python3射线法判断点是否在多边形内
2019/06/28 Python
如何基于python操作excel并获取内容
2019/12/24 Python
6行Python代码实现进度条效果(Progress、tqdm、alive-progress​​​​​​​和PySimpleGUI库)
2020/01/06 Python
Python 动态变量名定义与调用方法
2020/02/09 Python
Django def clean()函数对表单中的数据进行验证操作
2020/07/09 Python
python代码实现猜拳小游戏
2020/11/30 Python
香港钟表珠宝首饰商城:OneMallTime网摩间
2016/10/14 全球购物
索尼巴西商店:Sony巴西
2019/06/21 全球购物
高考备战决心书
2014/03/11 职场文书
2014年财务个人工作总结
2014/12/08 职场文书
商务代表岗位职责
2015/02/15 职场文书
2015年征兵工作总结
2015/07/23 职场文书
vue使用element-ui按需引入
2022/05/20 Vue.js
git stash(储藏)的用法总结
2022/06/25 Servers