3

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

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

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

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

/**
     * 微信企业付款
     * @param  [type] $openid   商户appid下,某用户的openid
     * @param  [type] $username 收款用户真实姓名。
     * @param  [type] $desc     企业付款操作说明信息。必填。
     * @param  [type] $money    企业付款金额,单位为分
     * @author appyjj <--1121099600@qq.com-->
     */
    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.'</br>';
        //签名步骤二:在string后加入KEY
        $String = $String."&key=1231231231234321abcdxingaaabb235";//此处请填写商户平台中的 API密钥
        //echo "【string2】".$String."</br>";
        //签名步骤三:MD5加密
        $String = md5($String);
        //echo "【string3】 ".$String."</br>";
        //签名步骤四:所有字符转为大写
        $result_ = strtoupper($String);
        //echo "【result】 ".$result_."</br>";
        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"

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

<xml>
<appid>wxd930ea5d5a258f4f</appid>
<mch_id>10000100</mch_id>
<device_info>1000<device_info>
<body>test</body>
<nonce_str>ibuaiVcKdpRxkhJA</nonce_str>
<sign>9A0A8659F005D6984697E2CA0A9CF3B7</sign>
<xml>
**以上示例摘抄自微信支付官方文档**

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

function createNoncestr( $length = 32 ) 
    {
        $chars = "abcdefghijklmnopqrstuvwxyz0123456789";  //这里不用修改,照抄即可
        $str ="";
        for ( $i = 0; $i < $length; $i++ )  {  
            $str.= substr($chars, mt_rand(0, strlen($chars)-1), 1);  
        }  
        return $str;
    }
太过简单,不做叙述!

数组转xml

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; 
    }

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"."<br>"; 
            curl_close($ch);
            return false;
        }
    }

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

图片描述

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

下面是完整代码示例

直接copy后,修改一下参数既可以使用!
<?php 
/**
 * 微信支付企业付款接口
 */
    ini_set('display_errors', 0); 
    header("Content-type:text/html;chartset=utf-8");
    $wxPay = new wxPay();
    $wxPay->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 <--1121099600@qq.com-->
     */
    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.'</br>';
        //签名步骤二:在string后加入KEY
        $String = $String."&key=guoyaojituanfengliaoxing82093235";
        //echo "【string2】".$String."</br>";
        //签名步骤三:MD5加密
        $String = md5($String);
        //echo "【string3】 ".$String."</br>";
        //签名步骤四:所有字符转为大写
        $result_ = strtoupper($String);
        //echo "【result】 ".$result_."</br>";
        return $result_;
    }

    /**
     *  作用:产生随机字符串,不长于32位
     */
    function createNoncestr( $length = 32 ) 
    {
        $chars = "abcdefghijklmnopqrstuvwxyz0123456789";  
        $str ="";
        for ( $i = 0; $i < $length; $i++ )  {  
            $str.= substr($chars, mt_rand(0, strlen($chars)-1), 1);  
        }  
        return $str;
    }

    /**
     *  作用:array转xml
     */
    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; 
    }

    /**
     *  作用:将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"."<br>"; 
            curl_close($ch);
            return false;
        }
    }
}
ok!此文半抄半写~总之代码没问题的~!分享创造动力,撸主继续干活了!

Leoappyjj
181 声望22 粉丝

PHP全栈工程师