前言

支付宝企业账户转账个人账户php接口代码,该接口主要用于平台对用户奖励的发放,场景也很简单就是我在你平台上有余额,我提现,平台给钱。

只不过项目中是用的官方的SDK调用的,仔细看了下那个SDK目录,把很多无用的接口(目前自己用不上)也包含在里面,作为强迫症的自己,想着精简一番。

注意

因为使用时间比较长,这里用的是转账的老接口,停止更新了,目前还能正常使用。

需要企业支付宝才能签约

收款账号必填,收款姓名可以不填,但是如果当账号对应多个名字的时候,会提示完善姓名,否则找不到哪个账号,我也不知道为啥一个账号会对应多个姓名这种,可能是支付宝前期账号和手机号没有处理好吧。

付款方(企业方)可以自定义显示的名称,比如企业名称太长想简化可以设置payer_show_name参数

当一个账号在某个很短的时间内,频繁提现相同金额,会触发支付宝的风控,因此 在调用该接口时一定要做好访问频率(同账号,同ip)限制。涉及到金钱的一定要把细节做好。

php代码

  1. <?php
  2. header(‘Content-type:text/html; Charset=utf-8’);
  3. // sdk 单笔转账接口  https://opendocs.alipay.com/apis/api_28/alipay.fund.trans.uni.transfer
  4. Class AliTransferPay
  5. {
  6.     public $appId;
  7.     public $privateKey;
  8.     public $charset ;
  9.     public $sign_type ;
  10.     public function __construct($appId, $privateKey)
  11.     {
  12.         $this->appId = $appId;
  13.         $this->privateKey = $privateKey;
  14.         $this->charset=‘utf-8’;
  15.         $this->sign_type=‘RSA2’;
  16.     }
  17.     /**
  18.     * 查询余额,需要签约改功能,如果不能签约的话 会提示 ISV权限不足,建议在开发者中心检查签约是否已经生效
  19.     * home.php?mod=space&uid=155549 mixed
  20.     */
  21.     public function queryBalance()
  22.     {
  23.         $reqParams = [
  24.             ‘app_id’ => $this->appId,
  25.             ‘method’ => ‘alipay.data.bill.balance.query’,
  26.             ‘format’ => ‘JSON’,
  27.             ‘charset’ => $this->charset,
  28.             ‘sign_type’ => $this->sign_type,
  29.             ‘timestamp’ => date(‘Y-m-d H:i:s’),
  30.             ‘version’ => ‘1.0’,
  31.         ];
  32.         //格式化编码
  33.         $bizParmsStr = $this->getSignContent($reqParams);
  34.         //计算签名
  35.         $sign = $this->getSign($bizParmsStr, $this->sign_type);
  36.         $reqParams[‘sign’] = $sign;
  37.         //发送请求
  38.         $result = $this->curlPost(‘https://openapi.alipay.com/gateway.do?charset=’.$this->charset, $reqParams);
  39.         return json_decode($result,true);
  40.     }
  41.     /**
  42.     * 转账方法
  43.     * home.php?mod=space&uid=952169 $order_id string 订单号
  44.     * @param $price string 价格
  45.     * @param $account string 收款方账户账号
  46.     * @param $payee_real_name string 收款方账户姓名 可以不传
  47.     * @param string $remark string 支付备注
  48.     * @param string $payer_show_name 转账企业支付时显示的名称
  49.     * @return mixed
  50.     */
  51.     public function pay($order_id, $price, $account, $payee_real_name, $remark = ‘测试’, $payer_show_name = ‘测试’)
  52.     {
  53.         $bizParms = [
  54.             ‘out_biz_no’ => $order_id,
  55.             ‘payee_type’ => ‘ALIPAY_LOGONID’,//固定值,老接口
  56.             ‘payee_account’ => $account,
  57.             ‘payee_real_name’ => $payee_real_name,  //收款方姓名(选填)
  58.             ‘amount’ => $price, //转账金额
  59.             ‘remark’ => $remark,  //转账备注(选填)
  60.             ‘payer_show_name’ => $payer_show_name,  //支付方名称,比如我不想显示公司名称太长了,可以自定义这个名称
  61.         ];
  62.         $reqParams = [
  63.             ‘app_id’ => $this->appId,
  64.             ‘method’ => ‘alipay.fund.trans.toaccount.transfer’,
  65.             ‘format’ => ‘JSON’,
  66.             ‘charset’ => $this->charset,
  67.             ‘sign_type’ => $this->sign_type,
  68.             ‘timestamp’ => date(‘Y-m-d H:i:s’),
  69.             ‘version’ => ‘1.0’,
  70.             ‘biz_content’ => json_encode($bizParms),
  71.         ];
  72.         //格式化编码
  73.         $bizParmsStr = $this->getSignContent($reqParams);
  74.         //计算签名
  75.         $sign = $this->getSign($bizParmsStr, $this->sign_type);
  76.         $reqParams[‘sign’] = $sign;
  77.         //发送请求
  78.         $result = $this->curlPost(‘https://openapi.alipay.com/gateway.do?charset=’.$this->charset, $reqParams);
  79.         return json_decode($result, true);
  80.     }
  81.     /**
  82.     * 根据加密类型,进行参数加密
  83.     * @param $params string
  84.     * @param string $sign_type 该函数只支持 RSA 和 RSA2两种方式
  85.     * @return string
  86.     */
  87.     public function getSign($params, $sign_type = “RSA”)
  88.     {
  89.         $privateKey = $this->privateKey;
  90.         $formatPrivateKey = “—–BEGIN RSA PRIVATE KEY—–\n” .
  91.         wordwrap($privateKey, 64, “\n”, true) .
  92.         “\n—–END RSA PRIVATE KEY—–“;
  93.         ($formatPrivateKey) or die(‘您使用的私钥格式错误,请检查RSA私钥配置’);
  94.         if ($sign_type == ‘RSA2’) {
  95.             openssl_sign($params, $sign, $formatPrivateKey, version_compare(PHP_VERSION, ‘5.4.0’, ‘<‘) ? SHA256 : OPENSSL_ALGO_SHA256); //OPENSSL_ALGO_SHA256是php5.4.8以上版本才支持
  96.         } else {
  97.             openssl_sign($params, $sign, $formatPrivateKey);
  98.         }
  99.         return base64_encode($sign);
  100.     }
  101.     /**
  102.     * 获取待签名的字符串
  103.     * @param $params
  104.     * @return string
  105.     */
  106.     public function getSignContent($params)
  107.     {
  108.         ksort($params);
  109.         $stringToBeSigned = “”;
  110.         $i = 0;
  111.         foreach ($params as $k => $v) {
  112.             if (false === $this->checkEmpty($v) && “@” != substr($v, 0, 1)) {
  113.                 // 转换成目标字符集
  114.                 $v = $this->transCoding($v, $this->charset);
  115.                 if ($i == 0) {
  116.                     $stringToBeSigned .= “$k” . “=” . “$v”;
  117.                 } else {
  118.                     $stringToBeSigned .= “&” . “$k” . “=” . “$v”;
  119.                 }
  120.                 $i++;
  121.             }
  122.         }
  123.         unset ($k, $v);
  124.         return $stringToBeSigned;
  125.     }
  126.     /**
  127.     * 转换字符集编码
  128.     * @param $data
  129.     * @param $targetCharset
  130.     * @return string
  131.     */
  132.     function transCoding($data, $targetCharset)
  133.     {
  134.         if (!empty($data)) {
  135.             $fileType = $this->charset;
  136.             if (strcasecmp($fileType, $targetCharset) != 0) {
  137.                 $data = mb_convert_encoding($data, $targetCharset, $fileType);
  138.             }
  139.         }
  140.         return $data;
  141.     }
  142.     /**
  143.     * 校验参数是否为空
  144.     * @param $value
  145.     * @return bool
  146.     */
  147.     protected function checkEmpty($value)
  148.     {
  149.         return empty($value) == true;
  150.     }
  151.     /**
  152.     * curl请求,注意https如果没有证书,需要设置不验证证书和host
  153.     * @param string $url
  154.     * @param string $postData
  155.     * @param array $options
  156.     * @param bool $isssl
  157.     * @return mixed
  158.     */
  159.     public function curlPost($url = , $postData = , $options = [], $isssl = true)
  160.     {
  161.         if (is_array($postData)) {
  162.             $postData = http_build_query($postData);
  163.         }
  164.         $ch = curl_init();
  165.         curl_setopt($ch, CURLOPT_URL, $url);
  166.         curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
  167.         curl_setopt($ch, CURLOPT_POST, 1);
  168.         curl_setopt($ch, CURLOPT_POSTFIELDS, $postData);
  169.         curl_setopt($ch, CURLOPT_TIMEOUT, 30);
  170.         if (!empty($options)) {
  171.             curl_setopt_array($ch, $options);
  172.         }
  173.         //https请求 不验证证书和host
  174.         if ($isssl) {
  175.             curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
  176.             curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
  177.         }
  178.         $data = curl_exec($ch);
  179.         curl_close($ch);
  180.         return $data;
  181.     }
  182. }
  183. //转账
  184. function invoke()
  185. {
  186.     //调用测试
  187.     $appid = ;
  188.     $order_id = uniqid();
  189.     $price = 0.1;
  190.     $saPrivateKey=;
  191.     $account = ;//收款方账号
  192.     $payee_real_name = ;//收款方真实姓名
  193.     $aliPay = new AliTransferPay($appid, $saPrivateKey);
  194.     $result = $aliPay->pay($order_id, $price, $account, $payee_real_name);
  195.     $result = $result[‘alipay_fund_trans_toaccount_transfer_response’];
  196.     if ($result[‘code’] && $result[‘code’] == ‘10000’) {
  197.         echo ‘转账成功’;
  198.     } else {
  199.         echo $result[‘msg’] . ‘ : ‘ . $result[‘sub_msg’];
  200.     }
  201. }
  202. invoke();