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下一个阿拉伯数字转中文数字的函数
Jul 16 PHP
PHP 创建标签云函数代码
May 26 PHP
浅析PHP 按位与或 (^ 、&amp;)
Jun 21 PHP
浅析memcache启动以及telnet命令详解
Jun 28 PHP
php inc文件使用的风险和注意事项
Nov 12 PHP
微信支付开发教程(一)微信支付URL配置
May 28 PHP
php5.3以后的版本连接sqlserver2000的方法
Jul 28 PHP
php实现改变图片直接打开为下载的方法
Apr 14 PHP
Symfony2框架学习笔记之表单用法详解
Mar 18 PHP
Zend Framework入门知识点小结
Mar 19 PHP
PHP的Json中文处理解决方案
Sep 29 PHP
php从身份证获取性别和出生年月
Feb 09 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
PHP定时更新程序设计思路分享
2014/06/10 PHP
解决laravel(5.5)访问public报错的问题
2019/10/12 PHP
通过PHP的Wrapper无缝迁移原有项目到新服务的实现方法
2020/04/02 PHP
jQuery 研究心得 取得属性的值
2007/11/30 Javascript
jQuery操作select的实例代码
2012/06/14 Javascript
JavaScript中对象属性的添加和删除示例
2014/05/12 Javascript
jquery点击改变class并toggle的实现代码
2016/05/15 Javascript
js修改onclick动作的四种方法(推荐)
2016/08/18 Javascript
Angular的$http的ajax的请求操作(推荐)
2017/01/10 Javascript
jQuery插件FusionCharts绘制2D环饼图效果示例【附demo源码】
2017/04/10 jQuery
seajs实现强制刷新本地缓存的方法分析
2017/10/16 Javascript
vue页面离开后执行函数的实例
2018/03/13 Javascript
Vue Echarts实现可视化世界地图代码实例
2019/05/07 Javascript
Node对CommonJS的模块规范
2019/11/06 Javascript
js基于canvas实现时钟组件
2021/02/07 Javascript
[01:00:10]完美世界DOTA2联赛PWL S2 FTD vs Inki 第二场 11.21
2020/11/24 DOTA
python使用电子邮件模块smtplib的方法
2016/08/28 Python
python cx_Oracle模块的安装和使用详细介绍
2017/02/13 Python
python opencv之SIFT算法示例
2018/02/24 Python
使用python对文件中的数值进行累加的实例
2018/11/28 Python
python 实现返回一个列表中出现次数最多的元素方法
2019/06/11 Python
django fernet fields字段加密实践详解
2019/08/12 Python
python GUI库图形界面开发之PyQt5窗口背景与不规则窗口实例
2020/02/25 Python
python对接ihuyi实现短信验证码发送
2020/05/10 Python
HTML5之SVG 2D入门8—文档结构及相关元素总结
2013/01/30 HTML / CSS
Myprotein法国官网:欧洲第一运动营养品牌
2019/03/26 全球购物
德国珠宝和配件商店:Styleserver
2021/02/23 全球购物
实习生单位鉴定意见
2013/12/04 职场文书
奥巴马演讲稿
2014/01/08 职场文书
2014年学校国庆主题活动方案
2014/09/16 职场文书
部门群众路线教育实践活动对照检查材料思想汇报
2014/10/07 职场文书
2014年市场部工作总结
2014/11/25 职场文书
公积金接收函格式
2015/01/30 职场文书
2015年行政人事部工作总结
2015/05/13 职场文书
Python import模块的缓存问题解决方案
2021/06/02 Python
一篇文章带你学习Mybatis-Plus(新手入门)
2021/08/02 Java/Android