網(wǎng)上的很多PHP微信掃碼支付接入教程都頗為復(fù)雜,且需要配置和引入較多的文件,本人通過整理后給出一個單文件版的,只有200行代碼,希望可以給各位想接入微信掃碼支付的帶來些許幫助和借鑒意義。
直接運(yùn)行該文件即可得到一個支付二維碼的圖片。
需要注意的事項(xiàng):
1.該文件需放到支付授權(quán)目錄下,可以在微信支付商戶平臺->產(chǎn)品中心->開發(fā)配置中設(shè)置。
2.如提示簽名錯誤可以通過微信支付簽名驗(yàn)證工具進(jìn)行驗(yàn)證:https://pay.weixin.qq.com/wiki/tools/signverify/
代碼如下:
<?php header('Content-type:text/html; Charset=utf-8'); $mchid = 'xxxxx'; $appid = 'xxxxx'; $apiKey = 'xxxxx'; $wxPay = new WxpayService($mchid,$appid,$apiKey); $outTradeNo = uniqid(); $payAmount = 0.01; $orderName = '支付測試'; $notifyUrl = 'https://www.xxx.com/wx/'; $payTime = time(); $arr = $wxPay->createJsBizPackage($payAmount,$outTradeNo,$orderName,$notifyUrl,$payTime); $url = 'http://qr.liantu.com/api.php?text='.$arr['code_url']; echo "<img src='{$url}' style='width:300px;'>"; class WxpayService { protected $mchid; protected $appid; protected $apiKey; public function __construct($mchid, $appid, $key) { $this->mchid = $mchid; $this->appid = $appid; $this->apiKey = $key;
} public function createJsBizPackage($totalFee, $outTradeNo, $orderName, $notifyUrl, $timestamp) { $config = array( 'mch_id' => $this->mchid, 'appid' => $this->appid, 'key' => $this->apiKey,
); $orderName = iconv('GBK','UTF-8',$orderName); $unified = array( 'appid' => $config['appid'], 'attach' => 'pay', 'body' => $orderName, 'mch_id' => $config['mch_id'], 'nonce_str' => self::createNonceStr(), 'notify_url' => $notifyUrl, 'out_trade_no' => $outTradeNo, 'spbill_create_ip' => '127.0.0.1', 'total_fee' => intval($totalFee * 100), 'trade_type' => 'NATIVE',
); $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 === false) { die('parse xml error');
} if ($unifiedOrder->return_code != 'SUCCESS') { die($unifiedOrder->return_msg);
} if ($unifiedOrder->result_code != 'SUCCESS') { die($unifiedOrder->err_code);
} $codeUrl = (array)($unifiedOrder->code_url); if(!$codeUrl[0]) exit('get code_url error'); $arr = array( "appId" => $config['appid'], "timeStamp" => $timestamp, "nonceStr" => self::createNonceStr(), "package" => "prepay_id=" . $unifiedOrder->prepay_id, "signType" => 'MD5', "code_url" => $codeUrl[0],
); $arr['paySign'] = self::getSign($arr, $config['key']); return $arr;
} public function notify() { $config = array( 'mch_id' => $this->mchid, 'appid' => $this->appid, 'key' => $this->apiKey,
); $postStr = $GLOBALS["HTTP_RAW_POST_DATA"]; $postObj = simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA); if ($postObj === false) { die('parse xml error');
} if ($postObj->return_code != 'SUCCESS') { die($postObj->return_msg);
} if ($postObj->result_code != 'SUCCESS') { die($postObj->err_code);
} $arr = (array)$postObj; unset($arr['sign']); if (self::getSign($arr, $config['key']) == $postObj->sign) { echo '<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>'; return $postObj;
}
} public static function curlGet($url = '', $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);
} 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 = '', $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); if (!empty($options)) {
curl_setopt_array($ch, $options);
} 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 && "null" != $v) { if ($urlEncode) { $v = urlencode($v);
} $buff .= $k . "=" . $v . "&";
}
} $reqPar = ''; if (strlen($buff) > 0) { $reqPar = substr($buff, 0, strlen($buff) - 1);
} return $reqPar;
}
}
本站文章版權(quán)歸原作者及原出處所有 。內(nèi)容為作者個人觀點(diǎn), 并不代表本站贊同其觀點(diǎn)和對其真實(shí)性負(fù)責(zé),本站只提供參考并不構(gòu)成任何投資及應(yīng)用建議。本站是一個個人學(xué)習(xí)交流的平臺,網(wǎng)站上部分文章為轉(zhuǎn)載,并不用于任何商業(yè)目的,我們已經(jīng)盡可能的對作者和來源進(jìn)行了通告,但是能力有限或疏忽,造成漏登,請及時聯(lián)系我們,我們將根據(jù)著作權(quán)人的要求,立即更正或者刪除有關(guān)內(nèi)容。本站擁有對此聲明的最終解釋權(quán)。