讀古今文學網 > 微信公眾平台開發:從零基礎到ThinkPHP5高性能框架實踐 > 17.3.1 模式一:靜態鏈接 >

17.3.1 模式一:靜態鏈接

掃碼支付模式一又稱靜態鏈接支付,開發前,商戶必須在公眾平台後台設置支付回調URL。該URL的功能是接收用戶掃碼後微信支付系統回調的product_id和openid。

掃碼支付的接口類定義如下。


 1 /**
 2  * 請求商家獲取商品信息接口
 3  */
 4 class NativeCall_pub extends Wxpay_server_pub
 5 {
 6     /**
 7      * 生成接口參數XML
 8      */
 9     function createXml
10     {
11         if($this->returnParameters["return_code"] == "SUCCESS"){
12             $this->returnParameters["appid"] = WxPayConf_pub::APPID;// 公眾賬號ID
13             $this->returnParameters["mch_id"] = WxPayConf_pub::MCHID;// 商戶號
14             $this->returnParameters["nonce_str"] = $this->createNoncestr;// 隨機字符串
15             $this->returnParameters["sign"] = $this->getSign($this->returnParameters);// 簽名
16         }
17         return $this->arrayToXml($this->returnParameters);
18     }
19 
20     /**
21      * 獲取product_id
22      */
23     function getProductId
24     {
25         $product_id = $this->data["product_id"];
26         return $product_id;
27     }
28 }
  

掃碼支付模式一生成二維碼的流程如下。

首先設置支付相關參數,其中需要自己指定的參數是產品ID。其他由系統自動獲取或自動生成。其代碼如下。


$this->parameters["appid"] = WxPayConf_pub::APPID;             // 公眾賬號ID
$this->parameters["mch_id"] = WxPayConf_pub::MCHID;            // 商戶號
$time_stamp = time;
$this->parameters["time_stamp"] = "$time_stamp";               // 時間戳
$this->parameters["nonce_str"] = $this->createNoncestr;   // 隨機字符串
$product_id = WxPayConf_pub::APPID."static";                      // 自定義商品ID
$nativeLink->setParameter("product_id","$product_id");         // 商品ID
  

生成之後,獲得的數組如下。


object(NativeLink_pub)[1]
    public 'parameters' => 
        array (size=5)
            'product_id' => string 'wxdbfd43c561acxxxxstatic' (length=24)
            'appid' => string 'wxdbfd43c561acxxxx' (length=18)
            'mch_id' => string '10012345' (length=8)
            'time_stamp' => string '1419733441' (length=10)
            'nonce_str' => string 'no6qegpf11rn13nyl2q9izsk60be7fxc' (length=32)
  

再使用簽名算法,將上述數據生成簽名,得到sign值,結果如下。


object(NativeLink_pub)[1]
    public 'parameters' => 
        array (size=6)
            'product_id' => string 'wxdbfd43c561acxxxxstatic' (length=24)
            'appid' => string 'wxdbfd43c561acxxxx' (length=18)
            'mch_id' => string '10012345' (length=8)
            'time_stamp' => string '1419733441' (length=10)
            'nonce_str' => string 'no6qegpf11rn13nyl2q9izsk60be7fxc' (length=32)
            'sign' => string '546CD81B0B66F57DC27BFEECEA1FB218' (length=32)
  

基於上述參數,將生成二維碼的鏈接地址,生成代碼如下。


// 獲取鏈接
$product_url = $nativeLink->getUrl;
  

生成二維碼的鏈接如下。


weixin:// wxpay/bizpayurl?appid=wxdbfd43c561acxxxx&mch_id=10012345&nonce_str=no6qe
gpf11rn13nyl2q9izsk60be7fxc&product_id=wxdbfd43c561acxxxxstatic&sign=546CD81B0B66F57DC27BFEECEA1FB218&time_stamp=1419733441
  

將上述鏈接使用二維碼生成接口,就可以生成一個模式一下的微信支付二維碼,如圖17-6所示。

圖17-6 微信支付模式一的二維碼

當用戶掃瞄上述支付二維碼時,回調接口URL將接收到來自微信服務器推送的靜態Native支付鏈接的通知,接收通知的代碼如下。


// 使用Native通知接口
$nativeCall = new NativeCall_pub;
// 接收微信請求
$xml = $GLOBALS['HTTP_RAW_POST_DATA'];
  

該代碼接收的XML通知數據如下。


<xml>
    <appid><![CDATA[wxdbfd43c561acxxxx]]></appid>
    <openid><![CDATA[oc-XIjh32OByBiak_gSZ6JOqGFx8]]
    ></openid>
    <mch_id><![CDATA[10012345]]></mch_id>
    <is_subscribe><![CDATA[Y]]></is_subscribe>
    <nonce_str><![CDATA[PvLH3nsJjQCvwnYY]]></nonce_str>
    <product_id><![CDATA[wxdbfd43c561acxxxxstatic]]></product_id>
    <sign><![CDATA[F1CBDE07E3B5AE6EAF4D4033368264EC]]></sign>
</xml>
  

統一支付將提取product_id參數的值,並填充其他支付參數,然後請求統一下單接口,代碼如下。


// 提取product_id
$product_id = $nativeCall->getProductId;

$unifiedOrder = new UnifiedOrder_pub;
$this->parameters["appid"] = WxPayConf_pub::APPID;        // 公眾賬號ID
$this->parameters["mch_id"] = WxPayConf_pub::MCHID;        // 商戶號
$this->parameters["spbill_create_ip"] = $_SERVER['REMOTE_ADDR'];        // 終端IP
$this->parameters["nonce_str"] = $this->createNoncestr;        // 隨機字符串
$this->parameters["sign"] = $this->getSign($this->parameters);        // 簽名

$unifiedOrder->setParameter("body","貢獻一分錢");        // 商品描述
// 自定義訂單號,此處僅作舉例
$timeStamp = time;
$out_trade_no = WxPayConf_pub::APPID."$timeStamp";
$unifiedOrder->setParameter("out_trade_no","$out_trade_no");        // 商戶訂單號 
$unifiedOrder->setParameter("total_fee","1");        // 總金額
$unifiedOrder->setParameter("notify_url",WxPayConf_pub::NOTIFY_URL);        // 通知地址
$unifiedOrder->setParameter("trade_type","NATIVE");        // 交易類型
$unifiedOrder->setParameter("product_id","$product_id");        // 用戶標識
// 非必填參數,商戶可根據實際情況選填
// $unifiedOrder->setParameter("sub_mch_id","XXXX");        // 子商戶號
// $unifiedOrder->setParameter("device_info","XXXX");        // 設備號
// $unifiedOrder->setParameter("attach","XXXX");        // 附加數據
// $unifiedOrder->setParameter("time_start","XXXX");        // 交易起始時間
// $unifiedOrder->setParameter("time_expire","XXXX");        // 交易結束時間
// $unifiedOrder->setParameter("goods_tag","XXXX");        // 商品標記 
// $unifiedOrder->setParameter("openid","XXXX");        // 用戶標識
// 獲取prepay_id
$prepay_id = $unifiedOrder->getPrepayId; 

統一支付將返回如下XML數據


<xml>
    <return_code><![CDATA[SUCCESS]]></return_code>
    <return_msg><![CDATA[OK]]></return_msg>
    <appid><![CDATA[wxdbfd43c561acxxxx]]></appid>
    <mch_id><![CDATA[10012345]]></mch_id>
    <nonce_str><![CDATA[JLQ67G1EhjfZvlKv]]></nonce_str>
    <sign><![CDATA[7A4F2751F955C32EB65063CC9E3EAB57]]></sign>
    <result_code><![CDATA[SUCCESS]]></result_code>
    <prepay_id><![CDATA[wx2014122820020936799023550244567827]]></prepay_id>
    <trade_type><![CDATA[NATIVE]]></trade_type>
    <code_url><![CDATA[weixin:// wxpay/bizpayurl?sr=yQtNpvo]]></code_url>
</xml>
  

上述數據中包含重要的prepay_id。提取出該參數,然後回調接口生成一個響應的XML數據,代碼如下。


// 設置返回碼
// 設置必填參數
// appid已填,商戶無須重複填寫
// mch_id已填,商戶無須重複填寫
// noncestr已填,商戶無須重複填寫
// sign已填,商戶無須重複填寫
$nativeCall->setReturnParameter("return_code","SUCCESS"); // 返回狀態碼
$nativeCall->setReturnParameter("result_code","SUCCESS"); // 業務結果
$nativeCall->setReturnParameter("prepay_id","$prepay_id");// 預支付ID

// 將結果返回微信
$returnXml = $nativeCall->returnXml;
echo $returnXml;
  

而生成的XML數據如下。


<xml>
    <return_code><![CDATA[SUCCESS]]></return_code>
    <result_code><![CDATA[SUCCESS]]></result_code>
    <prepay_id><![CDATA[wx2014122820020936799023550244567827]]></prepay_id>
    <appid><![CDATA[wxdbfd43c561acxxxx]]></appid>
    <mch_id>10012345</mch_id>
    <nonce_str><![CDATA[e2bpc9fz3ykc2tcpipyvnb1l2qf8my3d]]></nonce_str>
    <sign><![CDATA[32C698EA795C0FBCDBCED622D1E01168]]></sign>
</xml>
  

這個XML數據回顯給微信服務器後,用戶的微信客戶端將會顯示出支付界面,如圖17-7所示。

圖17-7 微信支付界面

當用戶點擊「立即支付」按鈕後,將會彈出輸入密碼插件,用戶輸入支付密碼後,一個支付過程就完成了。