评论

微信支付之H5支付

微信支付之公众号支付,单文件版,一个PHP文件搞定微信h5支付。

<?php
/**
 * H5支付
 *
 * 常见错误:
 * 1.网络环境未能通过安全验证,请稍后再试(原因:终端IP(spbill_create_ip)与用户实际调起支付时微信侧检测到的终端IP不一致)
 * 2.商家参数格式有误,请联系商家解决(原因:当前调起H5支付的referer为空)
 * 3.商家存在未配置的参数,请联系商家解决(原因:当前调起H5支付的域名与申请H5支付时提交的授权域名不一致)
 * 4.支付请求已失效,请重新发起支付(原因:有效期为5分钟,如超时请重新发起支付)
 * 5.请在微信外打开订单,进行支付(原因:H5支付不能直接在微信客户端内调起)
 */
header('Content-type:text/html; Charset=utf-8');
/** 请填写以下配置信息 */
$mchid = 'xxxxx';          //微信支付商户号 PartnerID 通过微信支付商户资料审核后邮件发送
$appid = 'xxxxx';  //微信支付申请对应的公众号的APPID
$appKey = 'xxxxx';   //微信支付申请对应的公众号的APP Key
$apiKey = 'xxxxx';   //https://pay.weixin.qq.com 帐户设置-安全设置-API安全-API密钥-设置API密钥
$outTradeNo = uniqid();     //你自己的商品订单号
$payAmount = 0.01;          //付款金额,单位:元
$orderName = '支付测试';    //订单标题
$notifyUrl = 'http://www.xxx.com/wx/notify.php';     //付款成功后的回调地址(不要有问号)
$returnUrl = 'http://www.baidu.com';     //付款成功后,页面跳转的地址
$wapUrl = 'www.xxx.com';   //WAP网站URL地址
$wapName = 'H5支付';       //WAP 网站名
/** 配置结束 */

$wxPay = new WxpayService($mchid,$appid,$apiKey);
$wxPay->setTotalFee($payAmount);
$wxPay->setOutTradeNo($outTradeNo);
$wxPay->setOrderName($orderName);
$wxPay->setNotifyUrl($notifyUrl);
$wxPay->setReturnUrl($returnUrl);
$wxPay->setWapUrl($wapUrl);
$wxPay->setWapName($wapName);

$mwebUrl= $wxPay->createJsBizPackage($payAmount,$outTradeNo,$orderName,$notifyUrl);
echo "<h1><a href='{$mwebUrl}'>点击跳转至支付页面</a></h1>";
exit();
class WxpayService
{
    protected $mchid;
    protected $appid;
    protected $apiKey;
    protected $totalFee;
    protected $outTradeNo;
    protected $orderName;
    protected $notifyUrl;
    protected $returnUrl;
    protected $wapUrl;
    protected $wapName;

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

    public function setTotalFee($totalFee)
    {
        $this->totalFee = $totalFee;
    }
    public function setOutTradeNo($outTradeNo)
    {
        $this->outTradeNo = $outTradeNo;
    }
    public function setOrderName($orderName)
    {
        $this->orderName = $orderName;
    }
    public function setWapUrl($wapUrl)
    {
        $this->wapUrl = $wapUrl;
    }
    public function setWapName($wapName)
    {
        $this->wapName = $wapName;
    }
    public function setNotifyUrl($notifyUrl)
    {
        $this->notifyUrl = $notifyUrl;
    }
    public function setReturnUrl($returnUrl)
    {
        $this->returnUrl = $returnUrl;
    }

    /**
     * 发起订单
     * @return array
     */
    public function createJsBizPackage()
    {
        $config = array(
            'mch_id' => $this->mchid,
            'appid' => $this->appid,
            'key' => $this->apiKey,
        );
        $scene_info = array(
            'h5_info' =>array(
                'type'=>'Wap',
                'wap_url'=>$this->wapUrl,
                'wap_name'=>$this->wapName,
            )
        );
        $unified = array(
            'appid' => $config['appid'],
            'attach' => 'pay',             //商家数据包,原样返回,如果填写中文,请注意转换为utf-8
            'body' => $this->orderName,
            'mch_id' => $config['mch_id'],
            'nonce_str' => self::createNonceStr(),
            'notify_url' => $this->notifyUrl,
            'out_trade_no' => $this->outTradeNo,
            'spbill_create_ip' => $_SERVER['REMOTE_ADDR'],
            'total_fee' => intval($this->totalFee * 100),       //单位 转为分
            'trade_type' => 'MWEB',
            'scene_info'=>json_encode($scene_info)
        );
        $unified['sign'] = self::getSign($unified, $config['key']);
        $responseXml = self::curlPost('https://api.mch.weixin.qq.com/pay/unifiedorder', self::arrayToXml($unified));
        $unifiedOrder = simplexml_load_string($responseXml, 'SimpleXMLElement', LIBXML_NOCDATA);
        if ($unifiedOrder->return_code != 'SUCCESS') {
            die($unifiedOrder->return_msg);
        }
        if($unifiedOrder->mweb_url){
            return $unifiedOrder->mweb_url.'&redirect_url='.urlencode($this->returnUrl);
        }
        exit('error');
    }
    public static function curlPost($url = '', $postData = '', $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 = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
        $str = '';
        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) {
                if ($urlEncode) {
                    $v = urlencode($v);
                }
                $buff .= $k . "=" . $v . "&";
            }
        }
        $reqPar = '';
        if (strlen($buff) > 0) {
            $reqPar = substr($buff, 0, strlen($buff) - 1);
        }
        return $reqPar;
    }
}

最后一次编辑于  2020-02-20  
点赞 3
收藏
评论

7 个评论

  • 八英里-
    八英里-
    2020-11-06

    微信 支付宝 等支付方式都可以帮忙申请接入哦 我们属于微信 支付宝的服务商 也和很多持牌三方支付公司和银行合作的 需要的朋友欢迎咨询 遇到任何支付限额 关闭交易权限问题也可以咨询 赵18320304957

    2020-11-06
    赞同
    回复
  • Mr.F
    Mr.F
    2020-08-11

    微信H5支付 只在手机百度极速版中 不好用 有没有解决办法 空白页

    2020-08-11
    赞同
    回复
  • 卢旭峰
    卢旭峰
    2020-05-23

    我有个问题请教:

    2.商家参数格式有误,请联系商家解决(原因:当前调起H5支付的referer为空)

    这里说的“当前调起H5支付的referer”是指PHP中通过curl发送的这个请求,还是只收到微信的支付URL在浏览器中发出的请求?如果是指后者,那应该如何设置referer呢?


    2020-05-23
    赞同
    回复 2
    • A0上善若水15592097610
      A0上善若水15592097610
      2020-11-21
      不错,这个评论功能
      2020-11-21
      回复
    • 乌纱帽
      乌纱帽
      2021-01-21
      js 设置在请求头中吧
      2021-01-21
      回复
  • Alexei🔅
    Alexei🔅
    2020-05-14

    appid是指服务号的appid吗,h5支付是否一定要认证服务号?

    2020-05-14
    赞同
    回复
  • Javen
    Javen
    2020-04-12

    Node.js 版本可以参考 此篇文章

    Java 版本可以参考 此篇文章

    文章评论区有源码链接

    • 同时支持 Api-v3 以及 Api-v2 版本接口
    • 同时支持多商户多应用
    • 通知支持国内与境外的普通商户模式和服务商模式,
    • v2 接口同时支持 MD5 以及 HMAC-SHA256 签名算法。

    如果对你有帮助请点击「有用」告知

    2020-04-12
    赞同
    回复
  • Zixu Liao
    Zixu Liao
    2020-02-24

    simplexml_load_string前为什么不判断一下是不是空值,有没有请求成功呢

    2020-02-24
    赞同
    回复
  • 东邪
    东邪
    2020-02-20

    有个小问题,只是空值不参与签名,"null"字符串还是可以参与签名的。

    2020-02-20
    赞同
    回复 1
    • 答题小程序
      答题小程序
      2020-02-20
      好的谢谢。
      2020-02-20
      回复
登录 后发表内容