新聞中心
這次總結(jié)一下用戶在微信內(nèi)打開網(wǎng)頁時(shí),可以調(diào)用微信支 付完成下單功能的模塊開發(fā),也就是在微信內(nèi)的H5頁面通過jsApi接口實(shí)現(xiàn)支付功能。當(dāng)然了,微信官網(wǎng)上的微信支付開發(fā)文檔也講解的很詳細(xì),并且有實(shí)現(xiàn) 代碼可供參考,有的朋友直接看文檔就可以自己實(shí)現(xiàn)此支付接口的開發(fā)了。

成都創(chuàng)新互聯(lián)-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價(jià)比庫車網(wǎng)站開發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫,直接使用。一站式庫車網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋庫車地區(qū)。費(fèi)用合理售后完善,十余年實(shí)體公司更值得信賴。
一、前言
為何我還寫一篇微信支付接口的博文呢?第一,我們 必須知道,所謂的工作經(jīng)驗(yàn)很多都是靠總結(jié)出來的,你只有總結(jié)了更多知識(shí),積累了更多經(jīng)驗(yàn),你才能在該行業(yè)中脫穎而出,我個(gè)人覺得如今的招聘,很多都需要工 作經(jīng)驗(yàn)(1年、3年、5年....),其實(shí),工作時(shí)間的長(zhǎng)久不能衡量一個(gè)人技術(shù)水平的高低,有的人一年的工作經(jīng)驗(yàn)?zāi)苣?年工作經(jīng)驗(yàn)的程序猿的工資,有的3 年工作經(jīng)驗(yàn)的卻有可能比別人只有一年工作經(jīng)驗(yàn)的還低,所以說,總結(jié)才能讓自己的知識(shí)體系,經(jīng)驗(yàn)深度更牛逼更穩(wěn)固(雖然寫一篇博文挺花費(fèi)時(shí)間的);第二,寫 博文分享給大家還是挺有成就感的,首先是能讓新手從我分享的博文中能學(xué)到東西,并且能快速將博文所講解的技術(shù)運(yùn)用到實(shí)際中來,所以我寫的博文基本上能讓新 人快速讀懂并且容易理解,另外,技術(shù)大神的話,看到博文有講解的不對(duì)之處,還可以指出,并且可以交流,何樂而不為呢,我們需要的就是分享和交流。
扯遠(yuǎn)了,直接進(jìn)入該主題的詳解。
現(xiàn)在的微信支付方式有N種,看下圖,有刷卡支付、 公眾號(hào)支付、掃碼支付和APP支付,另外還有支付工具的開發(fā),本博文選擇的是公眾號(hào)支付借口而開發(fā)進(jìn)行講解,其他幾種支付接口開發(fā)基本上思路都是一樣的, 只要你能看懂我這博文所講解的基本思路,你基本上也能獨(dú)自開發(fā)其他幾個(gè)支付接口。
二、思路詳解
我們可以拿微信支付接口文檔里的業(yè)務(wù)流程時(shí)序圖看 看,如下圖,基本思路是這樣子:首先在后臺(tái)生成一個(gè)鏈接,展示給用戶讓用戶點(diǎn)擊(例如頁面上有微信支付的按鈕),用戶點(diǎn)擊按鈕后,網(wǎng)站后臺(tái)會(huì)根據(jù)訂單的相 關(guān)信息生成一個(gè)支付訂單,此時(shí)會(huì)調(diào)用統(tǒng)一下單接口,對(duì)微信支付系統(tǒng)發(fā)起請(qǐng)求,而微信支付系統(tǒng)受到請(qǐng)求后,會(huì)根據(jù)請(qǐng)求過來的數(shù)據(jù),生成一個(gè) 預(yù)支付交易會(huì)話標(biāo)識(shí)(prepay_id,就是通過這個(gè)來識(shí)別該訂單的),我們的網(wǎng)站收到微信支付系統(tǒng)的響應(yīng)后,會(huì)得到prepay_id,然后通過自己 構(gòu)造微信支付所需要的參數(shù),接著將支付所需參數(shù)返回給客戶端,用戶此時(shí)可能會(huì)有一個(gè)訂單信息頁,會(huì)有一個(gè)按鈕,點(diǎn)擊支付,此時(shí)會(huì)調(diào)用JSAPI接口對(duì)微信 支付系統(tǒng)發(fā)起 請(qǐng)求支付,微信支付系統(tǒng)檢查了請(qǐng)求的相關(guān)合法性之后,就會(huì)提示輸入密碼,用戶此時(shí)輸入密碼確認(rèn),微信支付系統(tǒng)會(huì)對(duì)其進(jìn)行驗(yàn)證,通過的話會(huì)返回支付結(jié)果,然 后微信跳轉(zhuǎn)會(huì)H5頁面,這其中有一步是異步通知網(wǎng)站支付結(jié)果,我們網(wǎng)站需要對(duì)此進(jìn)行處理(比如說異步支付結(jié)果通過后,需要更新數(shù)據(jù)表或者訂單信息,例如標(biāo) 志用戶已支付該訂單了,同時(shí)也需要更新訂單日志,防止用戶重復(fù)提交訂單)。
#p#
三、代碼講解
本次開發(fā)環(huán)境用的是php5.6 + MySQL + Redis + Linux + Apache,所選用的框架的CI框架(這些環(huán)境不一定需要和我的一致,框架也可以自己選擇,反正自己稍微修改下代碼就能移植過去了)。
微信支付接口的開發(fā)代碼我已經(jīng)提前寫好了,在這里我對(duì)其進(jìn)行分析講解,方便大家能輕松理解,當(dāng)然,假如你有一定的基礎(chǔ),直接看代碼就能理清所有流程了,并且我的代碼基本上都寫上了注釋(對(duì)于新手來說,這一點(diǎn)比微信文檔所提供的代碼好一點(diǎn))。
1、構(gòu)造一個(gè)鏈接展示給用戶
這里我們提前需要知道一個(gè)點(diǎn),那就是請(qǐng)求統(tǒng)一下單接口需要微信用戶的openid(詳情可看這https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1),而獲取openid需要先獲取code(詳情可看這微信登錄接口),所以我們需要構(gòu)造一個(gè)獲取code的URL:
- Wxpay.php文件:
- defined('BASEPATH') OR exit('No direct script access allowed');
- class Wxpay extends MY_Controller {
- public function __construct() {
- parent::__construct();
- $this->load->model('wxpay_model');
- //$this->load->model('wxpay');
- }
- public function index() {
- //微信支付
- $this->smarty['wxPayUrl'] = $this->wxpay_model->retWxPayUrl();
- $this->displayView('wxpay/index.tpl');
- }
- }
在這先看看model里所寫的幾個(gè)類:model里有幾個(gè)類:微信支付類、統(tǒng)一下單接口類、響應(yīng)型接口基類、請(qǐng)求型接口基類、所有接口基類、配置類。為何要分那么多類而不在一個(gè)類里實(shí)現(xiàn)所有的方法的,因?yàn)?,這樣看起來代碼邏輯清晰,哪個(gè)類該干嘛就干嘛。
這里我直接附上model的代碼了,里面基本上每一個(gè)類每一個(gè)方法甚至每一行代碼都會(huì)有解釋的了,這里我就不對(duì)其展開一句句分析了:
- defined('BASEPATH') OR exit('No direct script access allowed');
- class Wxpay_model extends CI_Model {
- public function __construct() {
- parent::__construct();
- }
- /**
- * 返回可以獲得微信code的URL (用以獲取openid)
- * @return [type] [description]
- */
- public function retWxPayUrl() {
- $jsApi = new JsApi_handle();
- return $jsApi->createOauthUrlForCode();
- }
- /**
- * 微信jsapi點(diǎn)擊支付
- * @param [type] $data [description]
- * @return [type] [description]
- */
- public function wxPayJsApi($data) {
- $jsApi = new JsApi_handle();
- //統(tǒng)一下單接口所需數(shù)據(jù)
- $payData = $this->returnData($data);
- //獲取code碼,用以獲取openid
- $code = $_GET['code'];
- $jsApi->setCode($code);
- //通過code獲取openid
- $openid = $jsApi->getOpenId();
- $unifiedOrderResult = null;
- if ($openid != null) {
- //取得統(tǒng)一下單接口返回的數(shù)據(jù)
- $unifiedOrderResult = $this->getResult($payData, 'JSAPI', $openid);
- //獲取訂單接口狀態(tài)
- $returnMessage = $this->returnMessage($unifiedOrder, 'prepay_id');
- if ($returnMessage['resultCode']) {
- $jsApi->setPrepayId($retuenMessage['resultField']);
- //取得wxjsapi接口所需要的數(shù)據(jù)
- $returnMessage['resultData'] = $jsApi->getParams();
- }
- return $returnMessage;
- }
- }
- /**
- * 統(tǒng)一下單接口所需要的數(shù)據(jù)
- * @param [type] $data [description]
- * @return [type] [description]
- */
- public function returnData($data) {
- $payData['sn'] = $data['sn'];
- $payData['body'] = $data['goods_name'];
- $payData['out_trade_no'] = $data['order_no'];
- $payData['total_fee'] = $data['fee'];
- $payData['attach'] = $data['attach'];
- return $payData;
- }
- /**
- * 返回統(tǒng)一下單接口結(jié)果 (參考https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=9_1)
- * @param [type] $payData [description]
- * @param [type] $trade_type [description]
- * @param [type] $openid [description]
- * @return [type] [description]
- */
- public function getResult($payData, $trade_type, $openid = null) {
- $unifiedOrder = new UnifiedOrder_handle();
- if ($opneid != null) {
- $unifiedOrder->setParam('openid', $openid);
- }
- $unifiedOrder->setParam('body', $payData['body']); //商品描述
- $unifiedOrder->setParam('out_trade_no', $payData['out_trade_no']); //商戶訂單號(hào)
- $unifiedOrder->setParam('total_fee', $payData['total_fee']); //總金額
- $unifiedOrder->setParam('attach', $payData['attach']); //附加數(shù)據(jù)
- $unifiedOrder->setParam('notify_url', base_url('/Wxpay/pay_callback'));//通知地址
- $unifiedOrder->setParam('trade_type', $trade_type); //交易類型
- //非必填參數(shù),商戶可根據(jù)實(shí)際情況選填
- //$unifiedOrder->setParam("sub_mch_id","XXXX");//子商戶號(hào)
- //$unifiedOrder->setParam("device_info","XXXX");//設(shè)備號(hào)
- //$unifiedOrder->setParam("time_start","XXXX");//交易起始時(shí)間
- //$unifiedOrder->setParam("time_expire","XXXX");//交易結(jié)束時(shí)間
- //$unifiedOrder->setParam("goods_tag","XXXX");//商品標(biāo)記
- //$unifiedOrder->setParam("product_id","XXXX");//商品ID
- return $unifiedOrder->getResult();
- }
- /**
- * 返回微信訂單狀態(tài)
- */
- public function returnMessage($unifiedOrderResult,$field){
- $arrMessage=array("resultCode"=>0,"resultType"=>"獲取錯(cuò)誤","resultMsg"=>"該字段為空");
- if($unifiedOrderResult==null){
- $arrMessage["resultType"]="未獲取權(quán)限";
- $arrMessage["resultMsg"]="請(qǐng)重新打開頁面";
- }elseif ($unifiedOrderResult["return_code"] == "FAIL")
- {
- $arrMessage["resultType"]="網(wǎng)絡(luò)錯(cuò)誤";
- $arrMessage["resultMsg"]=$unifiedOrderResult['return_msg'];
- }
- elseif($unifiedOrderResult["result_code"] == "FAIL")
- {
- $arrMessage["resultType"]="訂單錯(cuò)誤";
- $arrMessage["resultMsg"]=$unifiedOrderResult['err_code_des'];
- }
- elseif($unifiedOrderResult[$field] != NULL)
- {
- $arrMessage["resultCode"]=1;
- $arrMessage["resultType"]="生成訂單";
- $arrMessage["resultMsg"]="OK";
- $arrMessage["resultField"] = $unifiedOrderResult[$field];
- }
- return $arrMessage;
- }
- /**
- * 微信回調(diào)接口返回 驗(yàn)證簽名并回應(yīng)微信
- * @param [type] $xml [description]
- * @return [type] [description]
- */
- public function wxPayNotify($xml) {
- $notify = new Wxpay_server();
- $notify->saveData($xml);
- //驗(yàn)證簽名,并回復(fù)微信
- //對(duì)后臺(tái)通知交互時(shí),如果微信收到商戶的應(yīng)答不是成功或者超時(shí),微信認(rèn)為通知失敗
- //微信會(huì)通過一定的策略(如30分鐘共8次),定期重新發(fā)起通知
- if ($notify->checkSign() == false) {
- $notify->setReturnParameter("return_code","FAIL");//返回狀態(tài)碼
- $notify->setReturnParameter("return_msg","簽名失敗");//返回信息
- } else {
- $notify->checkSign=TRUE;
- $notify->setReturnParameter("return_code","SUCCESS");//設(shè)置返回碼
- }
- return $notify;
- }
- }
- /**
- * JSAPI支付——H5網(wǎng)頁端調(diào)起支付接口
- */
- class JsApi_handle extends JsApi_common {
- public $code;//code碼,用以獲取openid
- public $openid;//用戶的openid
- public $parameters;//jsapi參數(shù),格式為json
- public $prepay_id;//使用統(tǒng)一支付接口得到的預(yù)支付id
- public $curl_timeout;//curl超時(shí)時(shí)間
- function __construct()
- {
- //設(shè)置curl超時(shí)時(shí)間
- $this->curl_timeout = WxPayConf::CURL_TIMEOUT;
- }
- /**
- * 生成獲取code的URL
- * @return [type] [description]
- */
- public function createOauthUrlForCode() {
- //重定向URL
- $redirectUrl = "http://www.itcen.cn/wxpay/confirm/".$orderId."?showwxpaytitle=1";
- $urlParams['appid'] = WxPayConf::APPID;
- $urlParams['redirect_uri'] = $redirectUrl;
- $urlParams['response_type'] = 'code';
- $urlParams['scope'] = 'snsapi_base';
- $urlParams['state'] = "STATE"."#wechat_redirect";
- //拼接字符串
- $queryString = $this->ToUrlParams($urlParams, false);
- return "https://open.weixin.qq.com/connect/oauth2/authorize?".$queryString;
- }
- /**
- * 設(shè)置code
- * @param [type] $code [description]
- */
- public function setCode($code) {
- $this->code = $code;
- }
- /**
- * 作用:設(shè)置prepay_id
- */
- public function setPrepayId($prepayId)
- {
- $this->prepay_id = $prepayId;
- }
- /**
- * 作用:獲取jsapi的參數(shù)
- */
- public function getParams()
- {
- $jsApiObj["appId"] = WxPayConf::APPID;
- $timeStamp = time();
- $jsApiObj["timeStamp"] = "$timeStamp";
- $jsApiObj["nonceStr"] = $this->createNoncestr();
- $jsApiObj["package"] = "prepay_id=$this->prepay_id";
- $jsApiObj["signType"] = "MD5";
- $jsApiObj["paySign"] = $this->getSign($jsApiObj);
- $this->parameters = json_encode($jsApiObj);
- return $this->parameters;
- }
- /**
- * 通過curl 向微信提交code 用以獲取openid
- * @return [type] [description]
- */
- public function getOpenId() {
- //創(chuàng)建openid 的鏈接
- $url = $this->createOauthUrlForOpenid();
- //初始化
- $ch = curl_init();
- curl_setopt($ch, CURL_TIMEOUT, $this->curl_timeout);
- curl_setopt($ch, CURL_URL, $url);
- curl_setopt($ch, CURL_SSL_VERIFYPEER, FALSE);
- curl_setopt($ch, CURL_SSL_VERIFYHOST, FALSE);
- curl_setopt($ch, CURL_HEADER, FALSE);
- curl_setopt($ch, CURL_RETURNTRANSFER, TRUE);
- //執(zhí)行curl
- $res = curl_exec($ch);
- curl_close($ch);
- //取出openid
- $data = json_decode($res);
- if (isset($data['openid'])) {
- $this->openid = $data['openid'];
- } else {
- return null;
- }
- return $this->openid;
- }
- /**
- * 生成可以獲取openid 的URL
- * @return [type] [description]
- */
- public function createOauthUrlForOpenid() {
- $urlParams['appid'] = WxPayConf::APPID;
- $urlParams['secret'] = WxPayConf::APPSECRET;
- $urlParams['code'] = $this->code;
- $urlParams['grant_type'] = "authorization_code";
- $queryString = $this->ToUrlParams($urlParams, false);
- return "https://api.weixin.qq.com/sns/oauth2/access_token?".$queryString;
- }
- }
- /**
- * 統(tǒng)一下單接口類
- */
- class UnifiedOrder_handle extends Wxpay_client_handle {
- public function __construct() {
- //設(shè)置接口鏈接
- $this->url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
- //設(shè)置curl超時(shí)時(shí)間
- $this->curl_timeout = WxPayConf::CURL_TIMEOUT;
- }
- }
- /**
- * 響應(yīng)型接口基類
- */
- class Wxpay_server_handle extends JsApi_common{
- public $data; //接收到的數(shù)據(jù),類型為關(guān)聯(lián)數(shù)組
- public $returnParams; //返回參數(shù),類型為關(guān)聯(lián)數(shù)組
- /**
- * 將微信請(qǐng)求的xml轉(zhuǎn)換成關(guān)聯(lián)數(shù)組
- * @param [type] $xml [description]
- * @return [type] [description]
- */
- public function saveData($xml) {
- $this->data = $this->xmlToArray($xml);
- }
- /**
- * 驗(yàn)證簽名
- * @return [type] [description]
- */
- public function checkSign() {
- $tmpData = $this->data;
- unset($temData['sign']);
- $sign = $this->getSign($tmpData);
- if ($this->data['sign'] == $sign) {
- return true;
- }
- return false;
- }
- /**
- * 設(shè)置返回微信的xml數(shù)據(jù)
- */
- function setReturnParameter($parameter, $parameterValue)
- {
- $this->returnParameters[$this->trimString($parameter)] = $this->trimString($parameterValue);
- }
- /**
- * 將xml數(shù)據(jù)返回微信
- */
- function returnXml()
- {
- $returnXml = $this->createXml();
- return $returnXml;
- }
- }
- /**
- * 請(qǐng)求型接口的基類
- */
- class Wxpay_client_handle extends JsApi_common{
- public $params; //請(qǐng)求參數(shù),類型為關(guān)聯(lián)數(shù)組
- public $response; //微信返回的響應(yīng)
- public $result; //返回參數(shù),類型類關(guān)聯(lián)數(shù)組
- public $url; //接口鏈接
- public $curl_timeout; //curl超時(shí)時(shí)間
- /**
- * 設(shè)置請(qǐng)求參數(shù)
- * @param [type] $param [description]
- * @param [type] $paramValue [description]
- */
- public function setParam($param, $paramValue) {
- $this->params[$this->tirmString($param)] = $this->trimString($paramValue);
- }
- /**
- * 獲取結(jié)果,默認(rèn)不使用證書
- * @return [type] [description]
- */
- public function getResult() {
- $this->postxml();
- $this->result = $this->xmlToArray($this->response);
- return $this->result;
- }
- /**
- * post請(qǐng)求xml
- * @return [type] [description]
- */
- public function postxml() {
- $xml = $this->createXml();
- $this->response = $this->postXmlCurl($xml, $this->curl, $this->curl_timeout);
- return $this->response;
- }
- public function createXml() {
- $this->params['appid'] = WxPayConf::APPID; //公眾號(hào)ID
- $this->params['mch_id'] = WxPayConf::MCHID; //商戶號(hào)
- $this->params['nonce_str'] = $this->createNoncestr(); //隨機(jī)字符串
- $this->params['sign'] = $this->getSign($this->params); //簽名
- return $this->arrayToXml($this->params);
- }
- }
- /**
- * 所有接口的基類
- */
- class JsApi_common {
- function __construct() {
- }
- public function trimString($value) {
- $ret = null;
- if (null != $value) {
- $ret = trim($value);
- if (strlen($ret) == 0) {
- $ret = null;
- }
- }
- return $ret;
- }
- /**
- * 產(chǎn)生隨機(jī)字符串,不長(zhǎng)于32位
- * @param integer $length [description]
- * @return [type] [description]
- */
- public function createNoncestr($length = 32) {
- $chars = "abcdefghijklmnopqrstuvwxyz0123456789";
- $str = '';
- for ($i = 0; $i < $length; $i++) {
- $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
- }
- return $str;
- }
- /**
- * 格式化參數(shù) 拼接字符串,簽名過程需要使用
- * @param [type] $urlParams [description]
- * @param [type] $needUrlencode [description]
- */
- public function ToUrlParams($urlParams, $needUrlencode) {
- $buff = "";
- ksort($urlParams);
- foreach ($urlParams as $k => $v) {
- if($needUrlencode) $v = urlencode($v);
- $buff .= $k .'='. $v .'&';
- }
- $reqString = '';
- if (strlen($buff) > 0) {
- $reqString = substr($buff, 0, strlen($buff) - 1);
- }
- return $reqString;
- }
- /**
- * 生成簽名
- * @param [type] $params [description]
- * @return [type] [description]
- */
- public function getSign($obj) {
- foreach ($obj as $k => $v) {
- $params[$k] = $v;
- }
- //簽名步驟一:按字典序排序參數(shù)
- ksort($params);
- $str = $this->ToUrlParams($params, false);
- //簽名步驟二:在$str后加入key
- $str = $str."$key=".WxPayConf::KEY;
- //簽名步驟三:md5加密
- $str = md5($str);
- //簽名步驟四:所有字符轉(zhuǎn)為大寫
- $result = strtoupper($str);
- return $result;
- }
- /**
- * array轉(zhuǎn)xml
- * @param [type] $arr [description]
- * @return [type] [description]
- */
- public function arrayToXml($arr) {
- $xml = "
"; - foreach ($arr as $k => $v) {
- if (is_numeric($val)) {
- $xml .= "<".$key.">".$key."".$key.">";
- } else {
- $xml .= "<".$key.">".$key.">";
- }
- }
- $xml .= "";
網(wǎng)頁名稱:基于H5的微信支付開發(fā)詳解
地址分享:http://www.dlmjj.cn/article/cogppjo.html


咨詢
建站咨詢
