ホームページ >バックエンド開発 >PHPチュートリアル >PHPがWeChatネイティブ決済(スキャンコード決済)機能を実装

PHPがWeChatネイティブ決済(スキャンコード決済)機能を実装

不言
不言オリジナル
2018-05-31 16:36:362217ブラウズ

この記事では主に WeChat ネイティブ決済とスキャン コード決済機能を実装するための PHP を詳しく紹介します。興味のある方は参考にしてください。

インターネット上の PHP WeChat スキャン コード決済アクセス チュートリアルは非常に複雑です。多くの設定とファイルの導入が必要であることを整理した後、わずか 200 行のコードを含む単一ファイルのバージョンを提供しました。これが、WeChat スキャン コード支払いにアクセスしたいすべての人に役立つことを願っています。

このファイルを直接実行して、支払い QR コードの画像を取得します。

注意事項:

1. このファイルは、WeChat Pay Merchant Platform -> Product Center -> Development Configuration で設定できる支払い認証ディレクトリに配置する必要があります。
2. 署名エラーが表示された場合は、WeChat 支払い署名検証ツールを使用して検証できます: WeChat パブリック プラットフォーム支払いインターフェイス デバッグ ツール

コードは次のとおりです:

<?php
header(&#39;Content-type:text/html; Charset=utf-8&#39;);
$mchid = &#39;xxxxx&#39;;     //微信支付商户号 PartnerID 通过微信支付商户资料审核后邮件发送
$appid = &#39;xxxxx&#39;; //公众号APPID 通过微信支付商户资料审核后邮件发送
$apiKey = &#39;xxxxx&#39;;  //https://pay.weixin.qq.com 帐户设置-安全设置-API安全-API密钥-设置API密钥
$wxPay = new WxpayService($mchid,$appid,$apiKey);
$outTradeNo = uniqid();   //你自己的商品订单号
$payAmount = 0.01;     //付款金额,单位:元
$orderName = &#39;支付测试&#39;;  //订单标题
$notifyUrl = &#39;https://www.xxx.com/wx/&#39;;   //付款成功后的回调地址(不要有问号)
$payTime = time();   //付款时间
$arr = $wxPay->createJsBizPackage($payAmount,$outTradeNo,$orderName,$notifyUrl,$payTime);
//生成二维码
$url = &#39;http://qr.liantu.com/api.php?text=&#39;.$arr[&#39;code_url&#39;];
echo "<img src=&#39;{$url}&#39; style=&#39;width:300px;&#39;>";

class WxpayService
{
  protected $mchid;
  protected $appid;
  protected $apiKey;

  public function __construct($mchid, $appid, $key)
  {
    $this->mchid = $mchid;
    $this->appid = $appid;
    $this->apiKey = $key;
  }

  /**
   * 发起订单
   * @param float $totalFee 收款总费用 单位元
   * @param string $outTradeNo 唯一的订单号
   * @param string $orderName 订单名称
   * @param string $notifyUrl 支付结果通知url 不要有问号
   * @param string $timestamp 订单发起时间
   * @return array
   */
  public function createJsBizPackage($totalFee, $outTradeNo, $orderName, $notifyUrl, $timestamp)
  {
    $config = array(
      &#39;mch_id&#39; => $this->mchid,
      &#39;appid&#39; => $this->appid,
      &#39;key&#39; => $this->apiKey,
    );
    $orderName = iconv(&#39;GBK&#39;,&#39;UTF-8&#39;,$orderName);
    $unified = array(
      &#39;appid&#39; => $config[&#39;appid&#39;],
      &#39;attach&#39; => &#39;pay&#39;,       //商家数据包,原样返回,如果填写中文,请注意转换为utf-8
      &#39;body&#39; => $orderName,
      &#39;mch_id&#39; => $config[&#39;mch_id&#39;],
      &#39;nonce_str&#39; => self::createNonceStr(),
      &#39;notify_url&#39; => $notifyUrl,
      &#39;out_trade_no&#39; => $outTradeNo,
      &#39;spbill_create_ip&#39; => &#39;127.0.0.1&#39;,
      &#39;total_fee&#39; => intval($totalFee * 100),    //单位 转为分
      &#39;trade_type&#39; => &#39;NATIVE&#39;,
    );
    $unified[&#39;sign&#39;] = self::getSign($unified, $config[&#39;key&#39;]);
    $responseXml = self::curlPost(&#39;https://api.mch.weixin.qq.com/pay/unifiedorder&#39;, self::arrayToXml($unified));
    $unifiedOrder = simplexml_load_string($responseXml, &#39;SimpleXMLElement&#39;, LIBXML_NOCDATA);
    if ($unifiedOrder === false) {
      die(&#39;parse xml error&#39;);
    }
    if ($unifiedOrder->return_code != &#39;SUCCESS&#39;) {
      die($unifiedOrder->return_msg);
    }
    if ($unifiedOrder->result_code != &#39;SUCCESS&#39;) {
      die($unifiedOrder->err_code);
    }
    $codeUrl = (array)($unifiedOrder->code_url);
    if(!$codeUrl[0]) exit(&#39;get code_url error&#39;);
    $arr = array(
      "appId" => $config[&#39;appid&#39;],
      "timeStamp" => $timestamp,
      "nonceStr" => self::createNonceStr(),
      "package" => "prepay_id=" . $unifiedOrder->prepay_id,
      "signType" => &#39;MD5&#39;,
      "code_url" => $codeUrl[0],
    );
    $arr[&#39;paySign&#39;] = self::getSign($arr, $config[&#39;key&#39;]);
    return $arr;
  }


  public function notify()
  {
    $config = array(
      &#39;mch_id&#39; => $this->mchid,
      &#39;appid&#39; => $this->appid,
      &#39;key&#39; => $this->apiKey,
    );
    $postStr = $GLOBALS["HTTP_RAW_POST_DATA"];

    $postObj = simplexml_load_string($postStr, &#39;SimpleXMLElement&#39;, LIBXML_NOCDATA);
    if ($postObj === false) {
      die(&#39;parse xml error&#39;);
    }
    if ($postObj->return_code != &#39;SUCCESS&#39;) {
      die($postObj->return_msg);
    }
    if ($postObj->result_code != &#39;SUCCESS&#39;) {
      die($postObj->err_code);
    }
    $arr = (array)$postObj;
    unset($arr[&#39;sign&#39;]);
    if (self::getSign($arr, $config[&#39;key&#39;]) == $postObj->sign) {
      echo &#39;<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>&#39;;
      return $postObj;
    }
  }

  /**
   * curl get
   *
   * @param string $url
   * @param array $options
   * @return mixed
   */
  public static function curlGet($url = &#39;&#39;, $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 static function curlPost($url = &#39;&#39;, $postData = &#39;&#39;, $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);
    $data = curl_exec($ch);
    curl_close($ch);
    return $data;
  }

  public static function createNonceStr($length = 16)
  {
    $chars = &#39;abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789&#39;;
    $str = &#39;&#39;;
    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 = &#39;&#39;;
    if (strlen($buff) > 0) {
      $reqPar = substr($buff, 0, strlen($buff) - 1);
    }
    return $reqPar;
  }
}

上記は内容全体ですこの記事をお読みいただきありがとうございます。詳細については、PHP 中国語 Web サイトを参照してください。

関連おすすめ:

php、WeChat決済の現金封緘を実現

php WeChat決済の公式アカウント決済機能

以上がPHPがWeChatネイティブ決済(スキャンコード決済)機能を実装の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。