ThinkPHP框架整合微信支付之JSAPI模式图文详解


Posted in PHP onApril 09, 2019

本文实例讲述了ThinkPHP框架整合微信支付之JSAPI模式。分享给大家供大家参考,具体如下:

目前微信是很火的,微信支付目前很少在网上能看到一系列详细的demo,因此,花一点时间来做一下关于微信支付系列教程,

本次教程是JSAPI模式支付,其他的还会继续写下去

首先,下载微信支付的demo,我们根据微信上的demo来整合到TP中。

介绍下我们这里需要用到的几个文件:
ThinkPHP框架整合微信支付之JSAPI模式图文详解

在demo文件夹中:

js_api_call.php:提供了微信jsapi的主要功能
log_.php:提供打印日志功能
notify_url.php:异步通知功能
notify_url.log:异步通知日志
qrcode.js:生成二维码js插件

接下来介绍下WxPayPubHelper文件夹下的文件:

ThinkPHP框架整合微信支付之JSAPI模式图文详解

cacert 文件夹是存放微信证书的(PS:具体我还没怎么用证书,虽然下载下来了,证书请在微信商户平台上下载)
SDKRuntimeException.php:这个就是处理异常的、
WxPay.pub.config.php:这个是做一些配置的,稍后会详细讲解
WxPayPubHelper.php:这个其实就是微信支付的工具类,对于初学者只要知道怎么用他里面的方法就够了

OK,了解了微信官方提供的文件,我们就可以开始整合到TP中了,废话不多说,这就开始!

step1:将demo中的WxPayPubHelper整个文件夹都复制到TP的Vendor目录下,像我这样:

ThinkPHP框架整合微信支付之JSAPI模式图文详解

step2:配置WxPay.pub.config.php文件:

ThinkPHP框架整合微信支付之JSAPI模式图文详解

这里的配置都有注释,如果还有不懂或者配置出现问题可以留言提问

同时我把微信的这个配置放到了TP的config中,这一步大家可以随意

<?php
return array(
 //'配置项'=>'配置值'
 define('WEB_HOST', '这是您的网站域名地址'),
 /*微信支付配置*/
 'WxPayConf_pub'=>array(
  'APPID' => '您的APPID',
  'MCHID' => '您的商户ID',
  'KEY' => '商户秘钥',
  'APPSECRET' => '您的APPSECRET',
  'JS_API_CALL_URL' => WEB_HOST.'/index.php/Home/WxJsAPI/jsApiCall',
  'SSLCERT_PATH' => WEB_HOST.'/ThinkPHP/Library/Vendor/WxPayPubHelper/cacert/apiclient_cert.pem',
  'SSLKEY_PATH' => WEB_HOST.'/ThinkPHP/Library/Vendor/WxPayPubHelper/cacert/apiclient_key.pem',
  'NOTIFY_URL' => WEB_HOST.'/index.php/Home/WxJsAPI/notify',
  'CURL_TIMEOUT' => 30
 )
);

step3:将生成二维码的js放在Public目录下(这里目前用不到,在用扫码支付的情况才用到这个js),将日志文件放在Public目录下:像我这样:

ThinkPHP框架整合微信支付之JSAPI模式图文详解

step4:创建控制器:这里创建了一个WxJsAPIController的控制器,这里大家随便起名字,只要这个跟你们在公众平台上的设置相对应就可以(公众平台设置稍后介绍)

ThinkPHP框架整合微信支付之JSAPI模式图文详解

下面是控制器的代码部分了,首先初始化控制器,将WxPayPubHelper导入

/**
  * 初始化
  */
 public function _initialize()
 {
  //引入WxPayPubHelper
  vendor('WxPayPubHelper.WxPayPubHelper');
 }

接下来是使用统一支付接口,获取prepay_id的方法:

public function jsApiCall()
 {
  //使用jsapi接口
  $jsApi = new \JsApi_pub();
  
  //=========步骤1:网页授权获取用户openid============
  //通过code获得openid
  if (!isset($_GET['code']))
  {
   //触发微信返回code码
   $url = $jsApi->createOauthUrlForCode(C('WxPayConf_pub.JS_API_CALL_URL'));
   Header("Location: $url");
  }else
  {
   //获取code码,以获取openid
   $code = $_GET['code'];
   $jsApi->setCode($code);
   $openid = $jsApi->getOpenId();
  }
  
  //=========步骤2:使用统一支付接口,获取prepay_id============
  //使用统一支付接口
  $unifiedOrder = new \UnifiedOrder_pub();
  
  //设置统一支付接口参数
  //设置必填参数
  //appid已填,商户无需重复填写
  //mch_id已填,商户无需重复填写
  //noncestr已填,商户无需重复填写
  //spbill_create_ip已填,商户无需重复填写
  //sign已填,商户无需重复填写
  $unifiedOrder->setParameter("openid",$openid);//商品描述
  $unifiedOrder->setParameter("body","贡献一分钱");//商品描述
  //自定义订单号,此处仅作举例
  $timeStamp = time();
  $out_trade_no = C('WxPayConf_pub.APPID').$timeStamp;
  $unifiedOrder->setParameter("out_trade_no",$out_trade_no);//商户订单号
  $unifiedOrder->setParameter("total_fee","1");//总金额
  $unifiedOrder->setParameter("notify_url",C('WxPayConf_pub.NOTIFY_URL'));//通知地址
  $unifiedOrder->setParameter("trade_type","JSAPI");//交易类型
  //非必填参数,商户可根据实际情况选填
  //$unifiedOrder->setParameter("sub_mch_id","XXXX");//子商户号
  //$unifiedOrder->setParameter("device_info","XXXX");//设备号
  //$unifiedOrder->setParameter("attach","XXXX");//附加数据
  //$unifiedOrder->setParameter("time_start","XXXX");//交易起始时间
  //$unifiedOrder->setParameter("time_expire","XXXX");//交易结束时间
  //$unifiedOrder->setParameter("goods_tag","XXXX");//商品标记
  //$unifiedOrder->setParameter("openid","XXXX");//用户标识
  //$unifiedOrder->setParameter("product_id","XXXX");//商品ID
  
  $prepay_id = $unifiedOrder->getPrepayId();
  //=========步骤3:使用jsapi调起支付============
  $jsApi->setPrepayId($prepay_id);
  
  $jsApiParameters = $jsApi->getParameters();
  
  $this->assign('jsApiParameters',$jsApiParameters);
  $this->display('pay');
  //echo $jsApiParameters;
 }

这里都是复制微信demo的,改改名字罢了,没什么其他的

接下来是异步通知方法,也是复制的微信demo上的

public function notify()
 {
  //使用通用通知接口
  $notify = new \Notify_pub();
  
  //存储微信的回调
  $xml = $GLOBALS['HTTP_RAW_POST_DATA'];
  $notify->saveData($xml);
  
  //验证签名,并回应微信。
  //对后台通知交互时,如果微信收到商户的应答不是成功或超时,微信认为通知失败,
  //微信会通过一定的策略(如30分钟共8次)定期重新发起通知,
  //尽可能提高通知的成功率,但微信不保证通知最终能成功。
  if($notify->checkSign() == FALSE){
   $notify->setReturnParameter("return_code","FAIL");//返回状态码
   $notify->setReturnParameter("return_msg","签名失败");//返回信息
  }else{
   $notify->setReturnParameter("return_code","SUCCESS");//设置返回码
  }
  $returnXml = $notify->returnXml();
  echo $returnXml;
  
  //==商户根据实际情况设置相应的处理流程,此处仅作举例=======
  
  //以log文件形式记录回调信息
//   $log_ = new Log_();
  $log_name= __ROOT__."/Public/notify_url.log";//log文件路径
  
  log_result($log_name,"【接收到的notify通知】:\n".$xml."\n");
  
  if($notify->checkSign() == TRUE)
  {
   if ($notify->data["return_code"] == "FAIL") {
    //此处应该更新一下订单状态,商户自行增删操作
    log_result($log_name,"【通信出错】:\n".$xml."\n");
   }
   elseif($notify->data["result_code"] == "FAIL"){
    //此处应该更新一下订单状态,商户自行增删操作
    log_result($log_name,"【业务出错】:\n".$xml."\n");
   }
   else{
    //此处应该更新一下订单状态,商户自行增删操作
    log_result($log_name,"【支付成功】:\n".$xml."\n");
   }
  
   //商户自行增加处理流程,
   //例如:更新订单状态
   //例如:数据库操作
   //例如:推送支付完成信息
  }
 }

这里我把记录日志的类写到了function.php中:

function log_result($file,$word)
{
 $fp = fopen($file,"a");
 flock($fp, LOCK_EX) ;
 fwrite($fp,"执行日期:".strftime("%Y-%m-%d-%H:%M:%S",time())."\n".$word."\n\n");
 flock($fp, LOCK_UN);
 fclose($fp);
}

好了 其实控制器的方法就这么多,没什么其他的了,下面看一下页面,直接上代码吧:

<!DOCTYPE html>
<html>
<head>
 <meta http-equiv="content-type" content="text/html;charset=utf-8"/>
 <title>微信安全支付</title>

 <script type="text/javascript">

  //调用微信JS api 支付
  function jsApiCall()
  {
   WeixinJSBridge.invoke(
    'getBrandWCPayRequest',
    <?php echo $jsApiParameters; ?>,
    function(res){
     WeixinJSBridge.log(res.err_msg);
     alert(res.err_code+res.err_desc+res.err_msg);
     //alert("{$jsApiParameters}");
    }
   );
  }

  function callpay()
  {
   if (typeof WeixinJSBridge == "undefined"){
    if( document.addEventListener ){
     document.addEventListener('WeixinJSBridgeReady', jsApiCall, false);
    }else if (document.attachEvent){
     document.attachEvent('WeixinJSBridgeReady', jsApiCall); 
     document.attachEvent('onWeixinJSBridgeReady', jsApiCall);
    }
   }else{
    jsApiCall();
   }
  }
 </script>
</head>
<body>
 </br></br></br></br>
 <div align="center">
  <button style="width:210px; height:30px; background-color:#FE6714; border:0px #FE6714 solid; cursor: pointer; color:white; font-size:16px;" type="button" onclick="callpay()" >贡献一下</button>
 </div>
</body>
</html>

无须改动什么,直接复制就好

接下来是微信公众平台上的配置了,这里我遇到过问题,如果有在这里遇到问题的同学请留言,比如出现了access_deined或者access_notallowed等问题,这都可能是因为这里配置不对。
请看配置过程截图:

ThinkPHP框架整合微信支付之JSAPI模式图文详解

点击修改进入配置:

ThinkPHP框架整合微信支付之JSAPI模式图文详解

好了,可以测试了:下面是我的测试截图:

用微信扫描二维码

ThinkPHP框架整合微信支付之JSAPI模式图文详解

微信上点击贡献一下出现支付页面:

ThinkPHP框架整合微信支付之JSAPI模式图文详解

到此为止,微信JSAPI支付功能就全部做好了

当然,如果你是第一次做,肯定会遇到各种问题,
如果你是新手,遇到的问题都不知道为什么,
及时你做过了再做我相信还是可能由于细节上的疏忽会出现问题
不过不要烦躁,耐心的去发现问题
有问题请留言,下面还会介绍微信扫码支付模式一,模式二的详细教程

微信支付教程扫码模式一:
//3water.com/article/159356.htm

微信支付教程扫码模式二:
//3water.com/article/159362.htm

微信支付教程刷卡支付:
//3water.com/article/159401.htm

希望本文所述对大家基于ThinkPHP框架的PHP程序设计有所帮助。

PHP 相关文章推荐
用PHP查询域名状态whois的类
Nov 25 PHP
swfupload 多文件上传实现代码
Aug 27 PHP
php将数据库中的电话号码读取出来并生成图片
Aug 31 PHP
php上传文件中文文件名乱码的解决方法
Nov 01 PHP
ThinkPHP实现将SESSION存入MYSQL的方法
Jul 22 PHP
Codeigniter(CI)框架分页函数及相关知识
Nov 03 PHP
PHP超牛逼无限极分类生成树方法
May 11 PHP
PHP消息队列用法实例分析
Feb 12 PHP
php数据访问之增删改查操作
May 09 PHP
php 伪造HTTP_REFERER页面URL来源的三种方法
Sep 22 PHP
php设计模式之职责链模式定义与用法经典示例
Sep 19 PHP
php 使用ActiveMQ发送消息,与处理消息操作示例
Feb 23 PHP
PHP7引入的&quot;??&quot;和&quot;?:&quot;的区别讲解
Apr 08 #PHP
PHP开发实现快递查询功能详解
Apr 08 #PHP
PHP中number_format()函数的用法讲解
Apr 08 #PHP
微信JSSDK分享功能图文实例详解
Apr 08 #PHP
ThinkPHP框架实现FTP图片上传功能示例
Apr 08 #PHP
详解php命令注入攻击
Apr 06 #PHP
PHP实现数组向任意位置插入,删除,替换数据操作示例
Apr 05 #PHP
You might like
兼容各大浏览器带关闭按钮的漂浮多组图片广告代码
2014/06/05 PHP
thinkphp浏览历史功能实现方法
2014/10/29 PHP
Yii rules常用规则示例
2016/03/15 PHP
老生常谈文本文件和二进制文件的区别
2017/02/27 PHP
MC Dialog js弹出层 完美兼容多浏览器(5.6更新)
2010/05/06 Javascript
JS 实现完美include载入实现代码
2010/08/05 Javascript
再论Javascript的类继承
2011/03/05 Javascript
基于jquery的二级联动菜单实现代码
2011/04/25 Javascript
jQuery + Flex 通过拖拽方式动态改变图片的代码
2011/08/03 Javascript
使用apply方法处理数组的三个技巧[译]
2012/09/20 Javascript
JS实现遮罩层效果的简单实例
2013/11/12 Javascript
动态创建script标签实现跨域资源访问的方法介绍
2014/02/28 Javascript
node.js中的path.resolve方法使用说明
2014/12/08 Javascript
使用jQuery获得内容以及内容的属性
2015/02/26 Javascript
在JavaScript中调用Java类和接口的方法
2016/09/07 Javascript
使用vue.js实现checkbox的全选和多个的删除功能
2017/02/17 Javascript
JS控件bootstrap suggest plugin使用方法详解
2017/03/25 Javascript
浅谈Vue.js 组件中的v-on绑定自定义事件理解
2017/11/17 Javascript
使用 vue.js 构建大型单页应用
2018/02/10 Javascript
bootstrap table实现iview固定列的效果实例代码详解
2019/09/30 Javascript
Vue根据条件添加click事件的方式
2019/11/09 Javascript
基于vue-simple-uploader封装文件分片上传、秒传及断点续传的全局上传插件功能
2021/02/23 Vue.js
编写自定义的Django模板加载器的简单示例
2015/07/21 Python
python中解析json格式文件的方法示例
2017/05/03 Python
python模拟事件触发机制详解
2018/01/19 Python
Python3之文件读写操作的实例讲解
2018/01/23 Python
在Python 2.7即将停止支持时,我们为你带来了一份python 3.x迁移指南
2018/01/30 Python
浅析Django 接收所有文件,前端展示文件(包括视频,文件,图片)ajax请求
2020/03/09 Python
python实现读取类别频数数据画水平条形图案例
2020/04/24 Python
Python select及selectors模块概念用法详解
2020/06/22 Python
教师自我剖析材料范文
2014/09/30 职场文书
教师党的群众路线教育实践活动学习心得体会
2014/10/30 职场文书
2014年审计人员工作总结
2014/12/19 职场文书
2016年小学端午节活动总结
2016/04/01 职场文书
基于Redis过期事件实现订单超时取消
2021/05/08 Redis
java调用Restful接口的三种方法
2021/08/23 Java/Android