映月读书网 > 微信公众平台开发:从零基础到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 微信支付界面

当用户点击“立即支付”按钮后,将会弹出输入密码插件,用户输入支付密码后,一个支付过程就完成了。