php实现微信支付之退款功能


Posted in PHP onMay 30, 2018

网上的很多PHP微信支付接入教程都颇为复杂,且需要配置和引入较多的文件,本人通过整理后给出一个单文件版的,希望可以给各位想接入微信支付的带来些许帮助和借鉴意义。

直接运行该文件即可给指定的微信用户退款。

需要注意的事项:

1.微信退款到零钱要求必传证书,需要到这里账户中心->账户设置->API安全->下载证书,然后修改代码中的证书路径
2.该文件需放到支付授权目录下,可以在微信支付商户平台->产品中心->开发配置中设置。
3.如提示签名错误可以通过微信支付签名验证工具进行验证:微信公众平台支付接口调试工具
4.错误码参照:参照地址

代码如下:

<?php
/**
 * 关于微信退款的说明
 * 1.微信退款要求必传证书,需要到https://pay.weixin.qq.com 账户中心->账户设置->API安全->下载证书,证书路径在第119行和122行修改
 * 2.错误码参照 :https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_4
 */
header('Content-type:text/html; Charset=utf-8');
$mchid = 'xxxxx';     //微信支付商户号 PartnerID 通过微信支付商户资料审核后邮件发送
$appid = 'xxxxx'; //微信支付申请对应的公众号的APPID
$apiKey = 'xxxxx';  //https://pay.weixin.qq.com 帐户设置-安全设置-API安全-API密钥-设置API密钥
$orderNo = '';           //商户订单号(商户订单号与微信订单号二选一,至少填一个)
$wxOrderNo = '';           //微信订单号(商户订单号与微信订单号二选一,至少填一个)
$totalFee = 0.01;          //订单金额,单位:元
$refundFee = 0.01;         //退款金额,单位:元
$refundNo = 'refund_'.uniqid();    //退款订单号(可随机生成)
$wxPay = new WxpayService($mchid,$appid,$apiKey);
$result = $wxPay->doRefund($totalFee, $refundFee, $refundNo, $wxOrderNo,$orderNo);
if($result===true){
  echo 'refund success';exit();
}
echo 'refund fail';
class WxpayService
{
  protected $mchid;
  protected $appid;
  protected $apiKey;
  public $data = null;
  public function __construct($mchid, $appid, $key)
  {
    $this->mchid = $mchid; //https://pay.weixin.qq.com 产品中心-开发配置-商户号
    $this->appid = $appid; //微信支付申请对应的公众号的APPID
    $this->apiKey = $key;  //https://pay.weixin.qq.com 帐户设置-安全设置-API安全-API密钥-设置API密钥
  }
  /**
   * 退款
   * @param float $totalFee 订单金额 单位元
   * @param float $refundFee 退款金额 单位元
   * @param string $refundNo 退款单号
   * @param string $wxOrderNo 微信订单号
   * @param string $orderNo 商户订单号
   * @return string
   */
  public function doRefund($totalFee, $refundFee, $refundNo, $wxOrderNo='',$orderNo='')
  {
    $config = array(
      'mch_id' => $this->mchid,
      'appid' => $this->appid,
      'key' => $this->apiKey,
    );
    $unified = array(
      'appid' => $config['appid'],
      'mch_id' => $config['mch_id'],
      'nonce_str' => self::createNonceStr(),
      'total_fee' => intval($totalFee * 100),    //订单金额  单位 转为分
      'refund_fee' => intval($refundFee * 100),    //退款金额 单位 转为分
      'sign_type' => 'MD5',      //签名类型 支持HMAC-SHA256和MD5,默认为MD5
      'transaction_id'=>$wxOrderNo,        //微信订单号
      'out_trade_no'=>$orderNo,    //商户订单号
      'out_refund_no'=>$refundNo,    //商户退款单号
      'refund_desc'=>'商品已售完',   //退款原因(选填)
    );
    $unified['sign'] = self::getSign($unified, $config['key']);
    $responseXml = $this->curlPost('https://api.mch.weixin.qq.com/secapi/pay/refund', self::arrayToXml($unified));
    $unifiedOrder = simplexml_load_string($responseXml, 'SimpleXMLElement', LIBXML_NOCDATA);
    if ($unifiedOrder === false) {
      die('parse xml error');
    }
    if ($unifiedOrder->return_code != 'SUCCESS') {
      die($unifiedOrder->return_msg);
    }
    if ($unifiedOrder->result_code != 'SUCCESS') {
      die($unifiedOrder->err_code);
    }
    return true;
  }
  public static function curlGet($url = '', $options = array())
  {
    $ch = curl_init($url);
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
    curl_setopt($ch, CURLOPT_TIMEOUT, 30);
    if (!empty($options)) {
      curl_setopt_array($ch, $options);
    }
    //https请求 不验证证书和host
    curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
    curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
    $data = curl_exec($ch);
    curl_close($ch);
    return $data;
  }
  public function curlPost($url = '', $postData = '', $options = array())
  {
    if (is_array($postData)) {
      $postData = http_build_query($postData);
    }
    $ch = curl_init();
    curl_setopt($ch, CURLOPT_URL, $url);
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
    curl_setopt($ch, CURLOPT_POST, 1);
    curl_setopt($ch, CURLOPT_POSTFIELDS, $postData);
    curl_setopt($ch, CURLOPT_TIMEOUT, 30); //设置cURL允许执行的最长秒数
    if (!empty($options)) {
      curl_setopt_array($ch, $options);
    }
    //https请求 不验证证书和host
    curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
    curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
    //第一种方法,cert 与 key 分别属于两个.pem文件
    //默认格式为PEM,可以注释
    curl_setopt($ch,CURLOPT_SSLCERTTYPE,'PEM');
    curl_setopt($ch,CURLOPT_SSLCERT,getcwd().'/cert/apiclient_cert.pem');
    //默认格式为PEM,可以注释
    curl_setopt($ch,CURLOPT_SSLKEYTYPE,'PEM');
    curl_setopt($ch,CURLOPT_SSLKEY,getcwd().'/cert/apiclient_key.pem');
    //第二种方式,两个文件合成一个.pem文件
//    curl_setopt($ch,CURLOPT_SSLCERT,getcwd().'/all.pem');
    $data = curl_exec($ch);
    curl_close($ch);
    return $data;
  }
  public static function createNonceStr($length = 16)
  {
    $chars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
    $str = '';
    for ($i = 0; $i < $length; $i++) {
      $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
    }
    return $str;
  }
  public static function arrayToXml($arr)
  {
    $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 static function getSign($params, $key)
  {
    ksort($params, SORT_STRING);
    $unSignParaString = self::formatQueryParaMap($params, false);
    $signStr = strtoupper(md5($unSignParaString . "&key=" . $key));
    return $signStr;
  }
  protected static function formatQueryParaMap($paraMap, $urlEncode = false)
  {
    $buff = "";
    ksort($paraMap);
    foreach ($paraMap as $k => $v) {
      if (null != $v && "null" != $v) {
        if ($urlEncode) {
          $v = urlencode($v);
        }
        $buff .= $k . "=" . $v . "&";
      }
    }
    $reqPar = '';
    if (strlen($buff) > 0) {
      $reqPar = substr($buff, 0, strlen($buff) - 1);
    }
    return $reqPar;
  }
}
?>

github下载地址

相关文章:

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

PHP 相关文章推荐
PHP 5.0对象模型深度探索之绑定
Sep 05 PHP
php session_start()关于Cannot send session cache limiter - headers already sent错误解决方法
Nov 27 PHP
PHP统计二维数组元素个数的方法
Nov 12 PHP
深入解读php中关于抽象(abstract)类和抽象方法的问题分析
Jan 03 PHP
使用php方法curl抓取AJAX异步内容思路分析及代码分享
Aug 25 PHP
Laravel 5框架学习之向视图传送数据(进阶篇)
Apr 08 PHP
PHP类的声明与实例化及构造方法与析构方法详解
Jan 26 PHP
php中让人头疼的浮点数运算分析
Oct 10 PHP
Centos 6.5下PHP 5.3安装ffmpeg扩展的步骤详解
Mar 02 PHP
PHP递归删除多维数组中的某个值
Apr 17 PHP
Laravel框架模型的创建及模型对数据操作示例
May 07 PHP
laravel框架中表单请求类型和CSRF防护实例分析
Nov 23 PHP
分享5个非常有用的Laravel Blade指令
May 30 #PHP
php实现微信支付之企业付款
May 30 #PHP
ThinkPHP5框架缓存查询操作分析
May 30 #PHP
PHP实现通过CURL上传文件功能示例
May 30 #PHP
php实现微信支付之现金红包
May 30 #PHP
php微信支付之公众号支付功能
May 30 #PHP
php实现微信原生支付(扫码支付)功能
May 30 #PHP
You might like
php面向对象全攻略 (十二) 抽象方法和抽象类
2009/09/30 PHP
php教程 插件机制在PHP中实现方案
2012/11/02 PHP
php中mt_rand()随机数函数用法
2014/11/24 PHP
ECSHOP在PHP5.5及高版本上报错的解决方法
2015/08/31 PHP
PHP模板引擎Smarty中变量的使用方法示例
2016/04/11 PHP
thinkPHP连接sqlite3数据库的实现方法(附Thinkphp代码生成器下载)
2016/05/27 PHP
javascript cookies 设置、读取、删除实例代码
2010/04/12 Javascript
javascript 保存文件到本地实现方法
2012/11/29 Javascript
去掉gridPanel表头全选框的小例子
2013/07/18 Javascript
javascript弹出层输入框(示例代码)
2013/12/11 Javascript
JavaScript制作windows经典扫雷小游戏
2015/03/31 Javascript
javascript学习笔记之函数定义
2015/06/25 Javascript
谈谈JavaScript中的几种借用方法
2016/08/09 Javascript
简单实现Bootstrap标签页
2020/08/09 Javascript
jquery点击回车键实现登录效果并默认焦点的方法
2018/03/09 jQuery
vue-router重定向不刷新问题的解决
2018/06/25 Javascript
vue 中 命名视图的用法实例详解
2019/08/14 Javascript
vue与django集成打包的实现方法
2019/11/11 Javascript
vue-router懒加载的3种方式汇总
2021/02/28 Vue.js
mac系统安装Python3初体验
2018/01/02 Python
python实现Zabbix-API监控
2018/09/17 Python
使用tensorflow实现VGG网络,训练mnist数据集方式
2020/05/26 Python
Python使用正则表达式实现爬虫数据抽取
2020/08/17 Python
理肤泉美国官网:La Roche-Posay
2018/01/17 全球购物
西安启天科技有限公司网络工程师面试题笔试题
2016/06/12 面试题
毕业生自荐书
2014/02/02 职场文书
决心书范文
2014/03/11 职场文书
检举信的格式及范文
2014/04/04 职场文书
病媒生物防治方案
2014/05/13 职场文书
2014年师德师风学习材料
2014/05/16 职场文书
公司财务会计主管应聘求职信
2014/09/26 职场文书
设立有限责任公司出资协议书
2014/11/01 职场文书
横空出世观后感
2015/06/09 职场文书
详解缓存穿透击穿雪崩解决方案
2021/05/28 Redis
JUnit5常用注解的使用
2021/07/02 Java/Android
php 文件上传至OSS及删除远程阿里云OSS文件
2021/07/04 PHP