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中路径问题的解决方案
Oct 09 PHP
PHP Header用于页面跳转要注意的几个问题总结
Oct 03 PHP
php 显示指定路径下的图片
Oct 29 PHP
php表单提交问题的解决方法
Apr 12 PHP
供参考的 php 学习提高路线分享
Oct 23 PHP
php读取富文本的时p标签会出现红线是怎么回事
May 13 PHP
PHP生成二维码的两个方法和实例
Jul 01 PHP
php 类自动载入的方法
Jun 03 PHP
Zend Framework入门教程之Zend_Config组件用法详解
Dec 09 PHP
PHP常量define和const的区别详解
May 18 PHP
PHP使用redis位图bitMap 实现签到功能
Oct 08 PHP
ThinkPHP5&amp;5.1实现验证码的生成、使用及点击刷新功能示例
Feb 07 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
人族 Terran 魔法与科技
2020/03/14 星际争霸
php中的时间处理
2006/10/09 PHP
PHP版国家代码、缩写查询函数代码
2011/08/14 PHP
php中常量DIRECTORY_SEPARATOR用法深入分析
2014/11/14 PHP
php中try catch捕获异常实例详解
2014/11/21 PHP
PHP使用trim函数去除字符串左右空格及特殊字符实例
2016/01/07 PHP
PHP getallheaders无法获取自定义头(headers)的问题
2016/03/23 PHP
JQuery 获取json数据$.getJSON方法的实例代码
2013/08/02 Javascript
javaScript的函数对象的声明详解
2015/02/06 Javascript
JS显示下拉列表框内全部元素的方法
2015/03/31 Javascript
基于jQuey实现鼠标滑过变色(整行变色)
2015/12/07 Javascript
JavaScript实现显示函数调用堆栈的方法
2016/04/21 Javascript
js实现碰撞检测特效代码分享
2016/10/16 Javascript
基于slideout.js实现移动端侧边栏滑动特效
2016/11/28 Javascript
js实现百度地图定位于地址逆解析,显示自己当前的地理位置
2016/12/08 Javascript
$.browser.msie 为空或不是对象问题的多种解决方法
2017/03/19 Javascript
js is_valid_filename验证文件名的函数
2017/07/19 Javascript
Vue组件通信实践记录(推荐)
2017/08/15 Javascript
vue+webpack 打包文件 404 页面空白的解决方法
2018/02/28 Javascript
vue富文本框(插入文本、图片、视频)的使用及问题小结
2018/08/17 Javascript
使用vue重构资讯页面的实例代码解析
2019/11/26 Javascript
vue3.0搭配.net core实现文件上传组件
2020/10/29 Javascript
全面理解Python中self的用法
2016/06/04 Python
全面了解python字符串和字典
2016/07/07 Python
深入解析python中的实例方法、类方法和静态方法
2019/03/11 Python
Python使用微信itchat接口实现查看自己微信的信息功能详解
2019/08/22 Python
Python HTMLTestRunner测试报告view按钮失效解决方案
2020/05/25 Python
关于探究python中sys.argv时遇到的问题详解
2021/02/23 Python
收藏!10个免费高清视频素材网站!【设计、视频剪辑必备】
2021/03/18 杂记
世界上最大的艺术和工艺用品商店:MisterArt.com
2018/07/13 全球购物
读群众路线心得体会
2014/03/07 职场文书
审计专业自荐信范文
2014/04/21 职场文书
向国旗敬礼活动小结
2014/09/27 职场文书
2015年大学迎新工作总结
2015/07/16 职场文书
详解redis分布式锁的这些坑
2021/05/19 Redis
vue点击弹窗自动触发点击事件的解决办法(模拟场景)
2021/05/25 Vue.js