当前位置:首页 > 文章列表 > 数据库 > MySQL > 微信退款接口demo,商户向用户转账,以零钱方式入账

微信退款接口demo,商户向用户转账,以零钱方式入账

来源:SegmentFault 2023-02-24 18:02:30 0浏览 收藏

来到golang学习网的大家,相信都是编程学习爱好者,希望在这里学习数据库相关编程知识。下面本篇文章就来带大家聊聊《微信退款接口demo,商户向用户转账,以零钱方式入账》,介绍一下MySQL、Linux、PHP、微信支付、laravel,希望对大家的知识积累有所帮助,助力实战开发!

微信商户向个人退款(转账)---零钱入账

本屌一向抱着开源精神的态度,为大家分享文章和技巧,当然~分享出来的代码都是可以直接copy in。。。。望大家看完后点赞打赏收藏!谢谢~!如果有写的不对或者描述不正确的,欢迎大家给我留言指出错误,我将立即修改,以免误导大众!

此篇文章如题,实现商户向个人支付,零钱入账。下面我还是老规矩,一行代码一行注释,请大家注意看,在文章最后会附上完整代码!~

定义函数,首先进行基本信息的定义

/**
     * 微信企业付款
     * @param  [type] $openid   商户appid下,某用户的openid
     * @param  [type] $username 收款用户真实姓名。
     * @param  [type] $desc     企业付款操作说明信息。必填。
     * @param  [type] $money    企业付款金额,单位为分
     * @author appyjj 
     */
    function pay($openid='', $username='', $desc='', $money=0){
        $apiUrl = $this->APPURL;//企业付款接口url
        $Parameters=array();
        $Parameters['amount']           = $money;//企业付款金额,单位为分
        $Parameters['check_name']       = 'NO_CHECK';//NO_CHECK:不校验真实姓名 FORCE_CHECK:强校验真实姓名(未实名认证的用户会校验失败,无法转账) OPTION_CHECK:针对已实名认证的用户才校验真实姓名(未实名认证用户不校验,可以转账成功)
        $Parameters['desc']             = $desc;//企业付款操作说明信息。必填。
        $Parameters['mch_appid']        = $this->APPID;//微信分配的公众账号ID
        $Parameters['mchid']            = $this->MCHID;//微信支付分配的商户号
        $Parameters['nonce_str']        = $this->createNoncestr();//随机字符串,不长于32位
        $Parameters['openid']           = $openid;//商户appid下,某用户的openid
        $Parameters['partner_trade_no'] = 'saso'.time().rand(10000, 99999);//商户订单号,需保持唯一性
        $Parameters['re_user_name']     = $username;//收款用户真实姓名。 如果check_name设置为FORCE_CHECK或OPTION_CHECK,则必填用户真实姓名
        $Parameters['spbill_create_ip'] = $_SERVER['SERVER_ADDR'];//调用接口的机器Ip地址
        $Parameters['sign']             = $this->getSign($Parameters);//签名
        $xml  = $this->arrayToXml($Parameters);
        $res  = $this->postXmlSSLCurl($xml,$apiUrl);
        $return = $this->xmlToArray($res);
        var_dump($return);
        //return $res;
    }

格式化参数,在签名过程中需要使用到

function formatBizQueryParaMap($paraMap, $urlencode)
    {
        $buff = "";
        ksort($paraMap);
        foreach ($paraMap as $k => $v)
        {
            if($urlencode)
            {
               $v = urlencode($v);
            }
            //$buff .= strtolower($k) . "=" . $v . "&";
            $buff .= $k . "=" . $v . "&";
        }
        $reqPar;
        if (strlen($buff) > 0) 
        {
            $reqPar = substr($buff, 0, strlen($buff)-1);
        }
        return $reqPar;
    }

此处代码很简单~不做过多注释。就一个排序、循环和字符串长度判断。

生成签名---这里很重要~撸主也弄了好一会~请大家务必认真看注释

function getSign($Obj)
    {
        foreach ($Obj as $k => $v)
        {
            $Parameters[$k] = $v;
        }
        //签名步骤一:按字典序排序参数 
        ksort($Parameters);
        $String = $this->formatBizQueryParaMap($Parameters, false);
        //echo '【string1】'.$String.'';
        //签名步骤二:在string后加入KEY
        $String = $String."&key=1231231231234321abcdxingaaabb235";//此处请填写商户平台中的 API密钥
        //echo "【string2】".$String."";
        //签名步骤三:MD5加密
        $String = md5($String);
        //echo "【string3】 ".$String."";
        //签名步骤四:所有字符转为大写
        $result_ = strtoupper($String);
        //echo "【result】 ".$result_."";
        return $result_;
    }

排序的规则,没理解的请看这:
◆ 参数名ASCII码从小到大排序(字典序);
◆ 如果参数的值为空不参与签名;
◆ 参数名区分大小写;
◆ 验证调用返回或微信主动通知签名时,传送的sign参数不参与签名,将生成的签名与该sign值作校验。
◆ 微信接口可能增加字段,验证签名时必须支持增加的扩展字段

第二步,在stringA最后拼接上key得到stringSignTemp字符串,并对stringSignTemp进行MD5运算,再将得到的字符串所有字符转换为大写,得到sign值signValue。

举例:

假设传送的参数如下:

appid:    wxd930ea5d5a258f4f
mch_id:    10000100
device_info:    1000
body:    test
nonce_str:    ibuaiVcKdpRxkhJA
第一步:对参数按照key=value的格式,并按照参数名ASCII字典序排序如下:
stringA="appid=wxd930ea5d5a258f4f&body=test&device_info=1000&mch_id=10000100&nonce_str=ibuaiVcKdpRxkhJA";

第二步:拼接API密钥:

stringSignTemp="stringA&key=192006250b4c09247ec02edce69f6a2d"
sign=MD5(stringSignTemp).toUpperCase()="9A0A8659F005D6984697E2CA0A9CF3B7"

最终得到最终发送的数据:

wxd930ea5d5a258f4f100001001000test
ibuaiVcKdpRxkhJA9A0A8659F005D6984697E2CA0A9CF3B7

**以上示例摘抄自微信支付官方文档**

产生随机字符串,长度不超过32位

function createNoncestr( $length = 32 ) 
    {
        $chars = "abcdefghijklmnopqrstuvwxyz0123456789";  //这里不用修改,照抄即可
        $str ="";
        for ( $i = 0; $i 

太过简单,不做叙述!

数组转xml

function arrayToXml($arr)
    {
        $xml = "";
        foreach ($arr as $key=>$val)
        {
             if (is_numeric($val))
             {
                $xml.="".$val."".$key.">"; 

             }
             else
                $xml.="".$key.">";  
        }
        $xml.="";
        return $xml; 
    }

xml转数组

function xmlToArray($xml)
    {       
        //将XML转为array        
        $array_data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);      
        return $array_data;
    }

好的,下面是最后一步,请求

使用证书,以POST的方式提交xml到对应接口的URL

function postXmlSSLCurl($xml,$url,$second=30)
    {
        $ch = curl_init();
        //超时时间
        curl_setopt($ch,CURLOPT_TIMEOUT,$second);
        //这里设置代理,如果有的话
        //curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
        //curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
        curl_setopt($ch,CURLOPT_URL, $url);
        curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
        curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE);
        //设置header
        curl_setopt($ch,CURLOPT_HEADER,FALSE);
        //要求结果为字符串且输出到屏幕上
        curl_setopt($ch,CURLOPT_RETURNTRANSFER,TRUE);
        //设置证书
        //使用证书:cert 与 key 分别属于两个.pem文件
        //默认格式为PEM,可以注释
        curl_setopt($ch,CURLOPT_SSLCERTTYPE,'PEM');
        curl_setopt($ch,CURLOPT_SSLCERT, $this->SSLCERT_PATH);
        //默认格式为PEM,可以注释
        curl_setopt($ch,CURLOPT_SSLKEYTYPE,'PEM');
        curl_setopt($ch,CURLOPT_SSLKEY, $this->SSLKEY_PATH);
        //post提交方式
        curl_setopt($ch,CURLOPT_POST, true);
        curl_setopt($ch,CURLOPT_POSTFIELDS,$xml);
        $data = curl_exec($ch);
        //返回结果
        if($data){
            curl_close($ch);
            return $data;
        }
        else { 
            $error = curl_errno($ch);
            echo "curl出错,错误码:$error"."
"; curl_close($ch); return false; } }

程序执行的时候往往没有一次性成功的~下面给大家附上返回的错误信息对照查询表

图片描述

更详细的信息~请参考微信官方说明文档:[传送门][2]

下面是完整代码示例

直接copy后,修改一下参数既可以使用!

pay('这里填写收款人的openid', '姓名', '这里是描述(如果报utf8的错误,请把这里改为数字,此参数最好是传值过来)', 100); //最后这里的100为金额  (单位为分:即100 = 1元)
class wxPay{
    //=======【证书路径设置】=====================================
    //证书路径,注意应该填写绝对路径
    protected $SSLCERT_PATH = 'cert/apiclient_cert.pem';//请各位大爷自己修改一下路径
    protected $SSLKEY_PATH =  'cert/apiclient_key.pem';//请各位大爷自己修改一下路径
    //=======【基本信息设置】=====================================
    //微信公众号身份的唯一标识。审核通过后,在微信发送的邮件中查看
    protected $APPID = 'wx123123123123';//填写您的appid。微信公众平台里的
    //受理商ID,身份标识
    protected $MCHID = '123123123';//商户id
    //商户支付密钥Key。审核通过后,在微信发送的邮件中查看
    protected $KEY = '192006250b4c09247ec02edce69f6a2d'; 
    //JSAPI接口中获取openid,审核后在公众平台开启开发模式后可查看
    protected $APPSECRET = '123123123123123123123123123123';
    //JSAPI接口地址
    protected $APPURL = 'https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers';
    
    /**
     * 微信企业付款
     * @param  [type] $openid   商户appid下,某用户的openid
     * @param  [type] $username 收款用户真实姓名。
     * @param  [type] $desc     企业付款操作说明信息。必填。
     * @param  [type] $money    企业付款金额,单位为分
     * @author appyjj 
     */
    function pay($openid='', $username='', $desc='', $money=0){
        $apiUrl = $this->APPURL;//企业付款接口url
        $Parameters=array();
        $Parameters['amount']           = $money;//企业付款金额,单位为分
        $Parameters['check_name']       = 'NO_CHECK';//NO_CHECK:不校验真实姓名 FORCE_CHECK:强校验真实姓名(未实名认证的用户会校验失败,无法转账) OPTION_CHECK:针对已实名认证的用户才校验真实姓名(未实名认证用户不校验,可以转账成功)
        $Parameters['desc']             = $desc;//企业付款操作说明信息。必填。
        $Parameters['mch_appid']        = $this->APPID;//微信分配的公众账号ID
        $Parameters['mchid']            = $this->MCHID;//微信支付分配的商户号
        $Parameters['nonce_str']        = $this->createNoncestr();//随机字符串,不长于32位
        $Parameters['openid']           = $openid;//商户appid下,某用户的openid
        $Parameters['partner_trade_no'] = 'saso'.time().rand(10000, 99999);//商户订单号,需保持唯一性
        $Parameters['re_user_name']     = $username;//收款用户真实姓名。 如果check_name设置为FORCE_CHECK或OPTION_CHECK,则必填用户真实姓名
        $Parameters['spbill_create_ip'] = $_SERVER['SERVER_ADDR'];//调用接口的机器Ip地址
        $Parameters['sign']             = $this->getSign($Parameters);//签名
        $xml  = $this->arrayToXml($Parameters);
        $res  = $this->postXmlSSLCurl($xml,$apiUrl);
        $return = $this->xmlToArray($res);
        var_dump($return);
        //return $res;
    }

    /**
     *  作用:格式化参数,签名过程需要使用
     */
    function formatBizQueryParaMap($paraMap, $urlencode)
    {
        $buff = "";
        ksort($paraMap);
        foreach ($paraMap as $k => $v)
        {
            if($urlencode)
            {
               $v = urlencode($v);
            }
            //$buff .= strtolower($k) . "=" . $v . "&";
            $buff .= $k . "=" . $v . "&";
        }
        $reqPar;
        if (strlen($buff) > 0) 
        {
            $reqPar = substr($buff, 0, strlen($buff)-1);
        }
        return $reqPar;
    }

    /**
     *  作用:生成签名
     */
    function getSign($Obj)
    {
        foreach ($Obj as $k => $v)
        {
            $Parameters[$k] = $v;
        }
        //签名步骤一:按字典序排序参数
        ksort($Parameters);
        $String = $this->formatBizQueryParaMap($Parameters, false);
        //echo '【string1】'.$String.'';
        //签名步骤二:在string后加入KEY
        $String = $String."&key=guoyaojituanfengliaoxing82093235";
        //echo "【string2】".$String."";
        //签名步骤三:MD5加密
        $String = md5($String);
        //echo "【string3】 ".$String."";
        //签名步骤四:所有字符转为大写
        $result_ = strtoupper($String);
        //echo "【result】 ".$result_."";
        return $result_;
    }

    /**
     *  作用:产生随机字符串,不长于32位
     */
    function createNoncestr( $length = 32 ) 
    {
        $chars = "abcdefghijklmnopqrstuvwxyz0123456789";  
        $str ="";
        for ( $i = 0; $i ";
        foreach ($arr as $key=>$val)
        {
             if (is_numeric($val))
             {
                $xml.="".$val."".$key.">"; 

             }
             else
                $xml.="".$key.">";  
        }
        $xml.="";
        return $xml; 
    }

    /**
     *  作用:将xml转为array
     */
    function xmlToArray($xml)
    {       
        //将XML转为array        
        $array_data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);      
        return $array_data;
    }

    /**
     *  作用:使用证书,以post方式提交xml到对应的接口url
     */
    function postXmlSSLCurl($xml,$url,$second=30)
    {
        $ch = curl_init();
        //超时时间
        curl_setopt($ch,CURLOPT_TIMEOUT,$second);
        //这里设置代理,如果有的话
        //curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
        //curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
        curl_setopt($ch,CURLOPT_URL, $url);
        curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
        curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE);
        //设置header
        curl_setopt($ch,CURLOPT_HEADER,FALSE);
        //要求结果为字符串且输出到屏幕上
        curl_setopt($ch,CURLOPT_RETURNTRANSFER,TRUE);
        //设置证书
        //使用证书:cert 与 key 分别属于两个.pem文件
        //默认格式为PEM,可以注释
        curl_setopt($ch,CURLOPT_SSLCERTTYPE,'PEM');
        curl_setopt($ch,CURLOPT_SSLCERT, $this->SSLCERT_PATH);
        //默认格式为PEM,可以注释
        curl_setopt($ch,CURLOPT_SSLKEYTYPE,'PEM');
        curl_setopt($ch,CURLOPT_SSLKEY, $this->SSLKEY_PATH);
        //post提交方式
        curl_setopt($ch,CURLOPT_POST, true);
        curl_setopt($ch,CURLOPT_POSTFIELDS,$xml);
        $data = curl_exec($ch);
        //返回结果
        if($data){
            curl_close($ch);
            return $data;
        }
        else { 
            $error = curl_errno($ch);
            echo "curl出错,错误码:$error"."
"; curl_close($ch); return false; } } }

ok!此文半抄半写~总之代码没问题的~!分享创造动力,撸主继续干活了!

文中关于mysql的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《微信退款接口demo,商户向用户转账,以零钱方式入账》文章吧,也可关注golang学习网公众号了解相关技术文章。

版本声明
本文转载于:SegmentFault 如有侵犯,请联系study_golang@163.com删除
自定义mysqli_result() 替代 mysql_result()自定义mysqli_result() 替代 mysql_result()
上一篇
自定义mysqli_result() 替代 mysql_result()
一张图看懂 数据集 JOIN
下一篇
一张图看懂 数据集 JOIN
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 笔灵AI生成答辩PPT:高效制作学术与职场PPT的利器
    笔灵AI生成答辩PPT
    探索笔灵AI生成答辩PPT的强大功能,快速制作高质量答辩PPT。精准内容提取、多样模板匹配、数据可视化、配套自述稿生成,让您的学术和职场展示更加专业与高效。
    14次使用
  • 知网AIGC检测服务系统:精准识别学术文本中的AI生成内容
    知网AIGC检测服务系统
    知网AIGC检测服务系统,专注于检测学术文本中的疑似AI生成内容。依托知网海量高质量文献资源,结合先进的“知识增强AIGC检测技术”,系统能够从语言模式和语义逻辑两方面精准识别AI生成内容,适用于学术研究、教育和企业领域,确保文本的真实性和原创性。
    22次使用
  • AIGC检测服务:AIbiye助力确保论文原创性
    AIGC检测-Aibiye
    AIbiye官网推出的AIGC检测服务,专注于检测ChatGPT、Gemini、Claude等AIGC工具生成的文本,帮助用户确保论文的原创性和学术规范。支持txt和doc(x)格式,检测范围为论文正文,提供高准确性和便捷的用户体验。
    30次使用
  • 易笔AI论文平台:快速生成高质量学术论文的利器
    易笔AI论文
    易笔AI论文平台提供自动写作、格式校对、查重检测等功能,支持多种学术领域的论文生成。价格优惠,界面友好,操作简便,适用于学术研究者、学生及论文辅导机构。
    40次使用
  • 笔启AI论文写作平台:多类型论文生成与多语言支持
    笔启AI论文写作平台
    笔启AI论文写作平台提供多类型论文生成服务,支持多语言写作,满足学术研究者、学生和职场人士的需求。平台采用AI 4.0版本,确保论文质量和原创性,并提供查重保障和隐私保护。
    35次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码