微信小程序支付功能 php后台对接完整代码分享


Posted in Javascript onJune 12, 2018

微信小程序支付,php后台对接完整代码,全是干货呀,拿过来可以直接使用。小程序在调起微信支付之前需要5个参数,这时候就需要携带code向后台请求,然后后台根据code获取openid 再进行服务器之间的。

一、准备工作

1、小程序注册,要以公司的以身份去注册一个小程序,才有微信支付权限;

2、绑定商户号。

3、在小程序填写合法域

 二、完成以上条件,你可以得到

     小程序appid 小程序秘钥    这两个用于获取用户openid;

     商户号id ,商户号秘钥     支付接口必须的;

三、开始开发

前台代码

/* 
 调起微信支付 
 @param 支付价格,不填写默认为1分钱 
*/ 
function pay(total_fee) { 
 
 var total_fee = total_fee; 
 wx.login({ 
 success: res => { 
 
 //code 用于获取openID的条件之一 
 var code = res.code; 
 wx.request({ 
 url: '后台地址/index.php', 
 method: "POST", 
 data: { 
  total_fee:total_fee, 
  code: code, 
 }, 
 header: { 
  'content-type': 'application/x-www-form-urlencoded' // 默认值 
 }, 
 success: function (res) { //后端返回的数据 
  var data = res.data; 
  console.log(data); 
  console.log(data["timeStamp"]); 
  wx.requestPayment({ 
  timeStamp: data['timeStamp'], 
  nonceStr: data['nonceStr'], 
  package: data['package'], 
  signType: data['signType'], 
  paySign: data['paySign'], 
  success: function (res) { 
  wx.showModal({ 
  title: '支付成功', 
  content: '', 
  }) 
  }, 
  fail: function (res) { 
  console.log(res); 
  } 
  }) 
 } 
 }); 
 
 
 } 
 }) 
 
}

以下是PHP后台代码 ,这里用的是tp框架

<?php 
namespace Home\Controller; 
use Think\Controller; 
class PayController extends Controller { 
 
 /** 
 * [callback 微信支付回调处理] 
 * @Author zhengmingzhou 
 * @DateTime 2018-05-22 
 * @return function [description] 
 */ 
 public function callback(){ 
 vendor("Wechart.WxPay.Api"); 
 vendor("Wechart.NativePay"); 
 vendor("Wechart.WxPay.Data"); 
 vendor("Wechart.WxPay.Notify"); 
 
 
 //获取微信返回支付信息 
 $xml = $GLOBALS['HTTP_RAW_POST_DATA']; 
 $WxPayData = new \WxPayDataBase(); 
 $result = $WxPayData->FromXml($xml); 
 if($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS'){ 
  //回调逻辑处理。。。 
  
  
  //返回给微信的参数 
  $data['RETURN_CODE'] = 'SUCCESS'; 
  $data['RETURN_MSG'] = 'OK'; 
 
 
 }else{ 
  $data['RETURN_CODE'] = 'FAIL'; 
  $data['RETURN_MSG'] = 'NO'; 
 } 
 //返回给微信 
 $xml = self::arrtoxml($data); 
 echo $xml; 
 } 
 
 /** 
 * [arrtoxml 格式化返回给微信的数据格式] 
 * @Author zhengmingzhou 
 * @DateTime 2018-05-22 
 * @param [type] $arr [description] 
 * @return [type]  [description] 
 */ 
 private function arrtoxml( $arr ){ 
 if(!$arr){ 
  return ''; 
 }else{ 
  $xml = "<xml>"; 
  foreach ($arr as $key=>$val) 
  { 
  if (is_numeric($val)){ 
   $xml.="<".$key.">".$val."</".$key.">"; 
  }else{ 
   $xml.="<".$key."><![CDATA[".$val."]]></".$key.">"; 
  } 
  } 
  $xml.="</xml>"; 
  return $xml; 
 } 
 } 
 
//微信支付 
public function pay(){ 
 //获取openid 
 if(I("post.code")) 
 { //用code获取openid 
  $code=I("post.code"); 
  $WX_APPID = '';//appid 
  $WX_SECRET = '';//AppSecret 
  $url = "https://api.weixin.qq.com/sns/jscode2session?appid=" . $WX_APPID . "&secret=" . $WX_SECRET . "&js_code=" . $code . "&grant_type=authorization_code"; 
  $infos = json_decode(file_get_contents($url)); 
  $openid = $infos->openid; 
 } 
 if(I("post.total_fee")) 
 { 
  $total_fee=I("post.total_fee"); 
 } 
 else 
 { 
  $total_fee=0.01; 
 } 
 
 $fee = 0.01;//举例充值0.01 
 $appid = '';//appid 
 $body =  '标题'; 
 $mch_id = ''; //商户号 
 $nonce_str = $this->nonce_str();//随机字符串 
 $notify_url = ''; //回调的url【自己填写】 
 $openid = $openid; 
 $out_trade_no = $this->order_number();//商户订单号 
 $spbill_create_ip = '';//服务器的ip【自己填写】; 
 $total_fee = $fee*100;//这里需要*100 
 $trade_type = 'JSAPI';//交易类型 默认 
 
 
 //这里是按照顺序的 因为下面的签名是按照顺序 排序错误 肯定出错 
 $post['appid'] = $appid; 
 $post['body'] = $body; 
 $post['mch_id'] = $mch_id; 
 $post['nonce_str'] = $nonce_str;//随机字符串 
 $post['notify_url'] = $notify_url; 
 $post['openid'] = $openid; 
 $post['out_trade_no'] = $out_trade_no; 
 $post['spbill_create_ip'] = $spbill_create_ip;//终端的ip 
 $post['total_fee'] = $total_fee;//总金额 
 $post['trade_type'] = $trade_type; 
 $sign = $this->sign($post);//签名 
 $post_xml = '<xml> 
  <appid>'.$appid.'</appid> 
  <body>'.$body.'</body> 
  <mch_id>'.$mch_id.'</mch_id> 
  <nonce_str>'.$nonce_str.'</nonce_str> 
  <notify_url>'.$notify_url.'</notify_url> 
  <openid>'.$openid.'</openid> 
  <out_trade_no>'.$out_trade_no.'</out_trade_no> 
  <spbill_create_ip>'.$spbill_create_ip.'</spbill_create_ip> 
  <total_fee>'.$total_fee.'</total_fee> 
  <trade_type>'.$trade_type.'</trade_type> 
  <sign>'.$sign.'</sign> 
 </xml> '; 
 
 
 //print_r($post_xml);die; 
 //统一接口prepay_id 
 $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder'; 
 $xml = $this->http_request($url,$post_xml); 
 
 $array = $this->xml($xml);//全要大写 
 
 
 //print_r($array); 
 if($array['RETURN_CODE'] == 'SUCCESS' && $array['RESULT_CODE'] == 'SUCCESS'){ 
 $time = time(); 
 $tmp='';//临时数组用于签名 
 $tmp['appId'] = $appid; 
 $tmp['nonceStr'] = $nonce_str; 
 $tmp['package'] = 'prepay_id='.$array['PREPAY_ID']; 
 $tmp['signType'] = 'MD5'; 
 $tmp['timeStamp'] = "$time"; 
 
 
 $data['state'] = 200; 
 $data['timeStamp'] = "$time";//时间戳 
 $data['nonceStr'] = $nonce_str;//随机字符串 
 $data['signType'] = 'MD5';//签名算法,暂支持 MD5 
 $data['package'] = 'prepay_id='.$array['PREPAY_ID'];//统一下单接口返回的 prepay_id 参数值,提交格式如:prepay_id=* 
 $data['paySign'] = $this->sign($tmp);//签名,具体签名方案参见微信公众号支付帮助文档; 
 $data['out_trade_no'] = $out_trade_no; 
 
 
 }else{ 
 $data['state'] = 0; 
 $data['text'] = "错误"; 
 $data['RETURN_CODE'] = $array['RETURN_CODE']; 
 $data['RETURN_MSG'] = $array['RETURN_MSG']; 
 } 
 
 
 echo json_encode($data); 
} 

 
//随机32位字符串 
private function nonce_str(){ 
 $result = ''; 
 $str = 'QWERTYUIOPASDFGHJKLZXVBNMqwertyuioplkjhgfdsamnbvcxz'; 
 for ($i=0;$i<32;$i++){ 
 $result .= $str[rand(0,48)]; 
 } 
 return $result; 
} 
 
 
//生成订单号 
private function order_number($openid){ 
 //date('Ymd',time()).time().rand(10,99);//18位 
 return md5($openid.time().rand(10,99));//32位 
} 
 
 
//签名 $data要先排好顺序 
private function sign($data){ 
 $stringA = ''; 
 foreach ($data as $key=>$value){ 
 if(!$value) continue; 
 if($stringA) $stringA .= '&'.$key."=".$value; 
 else $stringA = $key."=".$value; 
 } 
 $wx_key = '';//申请支付后有给予一个商户账号和密码,登陆后自己设置的key 
 $stringSignTemp = $stringA.'&key='.$wx_key; 
 return strtoupper(md5($stringSignTemp)); 
} 
 
 
//curl请求 
public function http_request($url,$data = null,$headers=array()) 
{ 
 $curl = curl_init(); 
 if( count($headers) >= 1 ){ 
 curl_setopt($curl, CURLOPT_HTTPHEADER, $headers); 
 } 
 curl_setopt($curl, CURLOPT_URL, $url); 
 
 
 curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, FALSE); 
 curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, FALSE); 
 
 
 if (!empty($data)){ 
 curl_setopt($curl, CURLOPT_POST, 1); 
 curl_setopt($curl, CURLOPT_POSTFIELDS, $data); 
 } 
 curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1); 
 $output = curl_exec($curl); 
 curl_close($curl); 
 return $output; 
} 
 
 
//获取xml 
private function xml($xml){ 
 $p = xml_parser_create(); 
 xml_parse_into_struct($p, $xml, $vals, $index); 
 xml_parser_free($p); 
 $data = ""; 
 foreach ($index as $key=>$value) { 
 if($key == 'xml' || $key == 'XML') continue; 
 $tag = $vals[$value[0]]['tag']; 
 $value = $vals[$value[0]]['value']; 
 $data[$tag] = $value; 
 } 
 return $data; 
} 
 
}

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

Javascript 相关文章推荐
JavaScript传递变量: 值传递?引用传递?
Feb 22 Javascript
jQuery选择器源码解读(四):tokenize方法的Expr.preFilter
Mar 31 Javascript
jQuery将表单序列化成一个Object对象的实例
Nov 29 Javascript
微信小程序中button组件的边框设置的实例详解
Sep 27 Javascript
vue 实现通过手机发送短信验证码注册功能
Apr 19 Javascript
详解create-react-app 2.0版本如何启用装饰器语法
Oct 23 Javascript
vue.js使用v-model实现表单元素(input) 双向数据绑定功能示例
Mar 08 Javascript
详解Vue之父子组件传值
Apr 01 Javascript
bootstrap tooltips在 angularJS中的使用方法
Apr 10 Javascript
JS 自执行函数原理及用法
Aug 05 Javascript
Vue内部渲染视图的方法
Sep 02 Javascript
JS中的继承操作实例总结
Jun 06 Javascript
js replace 全局替换的操作方法
Jun 12 #Javascript
微信小程序自定义prompt组件步骤详解
Jun 12 #Javascript
js实现购物车功能
Jun 12 #Javascript
浅谈Node.js 中间件模式
Jun 12 #Javascript
浅谈Webpack打包优化技巧
Jun 12 #Javascript
关于TypeScript模块导入的那些事
Jun 12 #Javascript
JS实现前端页面的搜索功能
Jun 12 #Javascript
You might like
5种PHP创建数组的实例代码分享
2014/01/17 PHP
删除html标签得到纯文本可处理嵌套的标签
2014/04/28 PHP
php无限级评论嵌套实现代码
2018/04/18 PHP
PHP折半(二分)查找算法实例分析
2018/05/12 PHP
PHP yield关键字功能与用法分析
2019/01/03 PHP
PHP _construct()函数讲解
2019/02/03 PHP
PHP数组与字符串互相转换实例
2020/05/05 PHP
JavaScript中的面向对象介绍
2012/06/30 Javascript
JScript分割字符串示例代码
2013/09/04 Javascript
使用简洁的jQuery方法实现隔行换色功能
2014/01/02 Javascript
js解析json读取List中的实体对象示例
2014/03/11 Javascript
运用jQuery定时器的原理实现banner图片切换
2014/10/22 Javascript
JavaScript 动态加载脚本和样式的方法
2015/04/13 Javascript
jQuery移动web开发之页面跳转和加载外部页面的实现
2015/12/04 Javascript
基于javascript实现tab切换特效
2016/03/29 Javascript
jQuery Easyui DataGrid点击某个单元格即进入编辑状态焦点移开后保存数据
2016/08/15 Javascript
JS代码实现百度地图 画圆 删除标注
2016/10/12 Javascript
详谈javascript精度问题与调整
2017/07/08 Javascript
vue实现可视化可拖放的自定义表单的示例代码
2019/03/20 Javascript
微信小程序第三方框架对比 之 wepy / mpvue / taro
2019/04/10 Javascript
JS常用排序方法实例代码解析
2020/03/03 Javascript
举例讲解Python程序与系统shell交互的方式
2015/04/09 Python
python实现用户登陆邮件通知的方法
2015/07/09 Python
Python如何实现守护进程的方法示例
2017/02/08 Python
python difflib模块示例讲解
2017/09/13 Python
推荐10款最受Python开发者欢迎的Python IDE
2018/09/16 Python
python从入门到精通 windows安装python图文教程
2019/05/18 Python
keras处理欠拟合和过拟合的实例讲解
2020/05/25 Python
Python使用OpenPyXL处理Excel表格
2020/07/02 Python
css3的transition效果和transfor效果示例介绍
2013/10/30 HTML / CSS
化工专业大学生职业生涯规划书
2014/01/14 职场文书
解除劳动关系协议书范文
2014/09/11 职场文书
2014院党委领导班子及其成员群众路线对照检查材料思想汇报
2014/10/04 职场文书
同学聚会祝酒词
2015/08/10 职场文书
总经理聘用协议书
2015/09/21 职场文书
微信小程序调用python模型
2022/04/21 Python