# 微信支付 實戰開發
> 現階段,最火的開發都離不開微信相關的
> 近期工作比較忙 正好工作正好是開發微信支付 , 那我現在就查一個微信支付的教程吧
> 希望對著手微信支付的人有所幫助,
微信支付,肯定離不開早一些相關成熟的微信支付類庫了,但我找了很久,完全適合TP5,并且符合相關命名規則的的類庫沒有找到.
> 那我就自己改一下吧 現在共享給大家.
> 文件放在 extend/com/wxpay文件夾下
> 命名空間問com\wxpay
> 源代碼我就放在本節微信支付類庫源代碼
接下來,我們一起開發微信支付吧
## 微信的支付流程
簡單的說一下支付的流程
第一步 獲取openid
第二步 根據openid進行統一下單獲取訂單號
第三步 根據統一下單的訂單號 獲取獲取JsApi$getParameters參數
最后 JsApi中支付
下面根據這個順路 我們一起來做吧
## 在base模塊建立一個公共的新控制器
> 我喜歡把一些公用的處理邏輯的類都放到base模塊,
> 
> 這些以后都是你的寶貝 .. 你懂得
建立好WxPay控制器以后,記得把接口文件的列表也挪過來
~~~
<?php
/**
* Created by PhpStorm.
* User: Mikkle
* Email:776329498@qq.com
* Date: 2017/2/4
* Time: 15:48
*/
namespace app\base\controller;
/**
* 微信支付幫助庫
* ====================================================
* 命名空間 com\wxpay\+類名+_pub
* 接口分三種類型:
* 【請求型接口】--Wxpay_client_
* 統一支付接口類--UnifiedOrder
* 訂單查詢接口--OrderQuery
* 退款申請接口--Refund
* 退款查詢接口--RefundQuery
* 對賬單接口--DownloadBill
* 短鏈接轉換接口--ShortUrl
* 【響應型接口】--Wxpay_server_
* 通用通知接口--Notify
* Native支付——請求商家獲取商品信息接口--NativeCall
* 【其他】
* 靜態鏈接二維碼--NativeLink
* JSAPI支付--JsApi
* =====================================================
* 【CommonUtil】常用工具:
* trimString(),設置參數時需要用到的字符處理函數
* createNoncestr(),產生隨機字符串,不長于32位
* formatBizQueryParaMap(),格式化參數,簽名過程需要用到
* getSign(),生成簽名
* arrayToXml(),array轉xml
* xmlToArray(),xml轉 array
* postXmlCurl(),以post方式提交xml到對應的接口url
* postXmlSSLCurl(),使用證書,以post方式提交xml到對應的接口url
*/
use think\Controller;
use think\Db;
use think\Session;
use com\wxpay\UnifiedOrder_pub as UnifiedOrder;
use com\wxpay\JsApi_pub as JaApi;
class WxPay extends Controller{
}
~~~
## 建立一個微信配置config參數
~~~
protected $wx_config=[
'wechat_appid'=>'wx1111111111111',//微信公眾號身份的唯一標識。審核通過后,在微信發送的郵件中查看
'wechat_mchid'=>'1111111111',//受理商ID,身份標識 商戶號
'wechat_appkey'=>'********************************',//商戶支付密鑰Key。審核通過后,在微信發送的郵件中查看
'wechat_appsecret'=>'****************************',//JSAPI接口中獲取openid,審核后在公眾平臺開啟開發模式后可查看
//證書路徑,注意應該填寫絕對路徑 不用證書也是能支付的
'sslcert_path'=>'',
'sslkey_path'=> '',
];
protected $this->notify_url=""; //自己定義
~~~
這些參數是你微信支付的參數
## 訂單數據庫的設置參數
為了以后程序移植方便,我把訂單數據庫的數據參數化了
這樣大家移植也方便多了
代碼如下
~~~
protected $order_table_param=[
'table'=>'my_orders', //訂單表名稱
'no_field'=>'order_no', //訂單號 字段名字
'state_field'=> 'is_pay',//訂單支付狀態值字段名
'amount_field'=>'amount',//訂單金額值字段名
'pay_ok'=> '1', //訂單已支付狀態值
'pay_no'=> '0', // 訂單未支付狀態值
'map' => [['status' => 1] ,[ 'order_state'=>0]], //其他訂單是否可以支付的參數值
];
~~~
## 接下來 類的初始化
在初始化里, 首先定義時區 ,這是參照官方的文檔的
接下來 動態加載config參數
我把支付的參數寫到這個文檔里,而沒有寫的config文件中,主要為了以后移植方便
接下來注入openid,我微信登錄是在控制器類就完成的 所以這里就直接獲取session
然后注入到$this->open_id
~~~
/**
* #User: Mikkle
* #Email:776329498@qq.com
* #Date:
*/
public function _initialize(){
ini_set('date.timezone','Asia/Shanghai');
config($this->wx_config);
//已登陸的設置openid 本人微信登錄是在控制器里完成
if (Session::has('open_id','html5')) $this->open_id=Session::get('open_id','html5');
}
~~~
## 統一下單方法
接下來 我把官方的統一下單封裝了一個方法 不多說了 源代碼如下
~~~
/**
* 統一下單方法
* #User: Mikkle
* #Email:776329498@qq.com
* #Date:
* @param array $data
* @return bool
*/
protected function unifiedOrder($data=[]){
$unifiedOrder = new UnifiedOrder();
$unifiedOrder->setParameter("openid",$this->open_id); // openid
$unifiedOrder->setParameter("body",'商品訂單號'+$data['order_no']); // 商品描術
$unifiedOrder->setParameter("out_trade_no",$data['order_no'].'_'.$unifiedOrder->createNoncestr(6)); // 商戶訂單號
$unifiedOrder->setParameter("total_fee",$data['amount']*100); // 總金額
$unifiedOrder->setParameter("notify_url",$this->notify_url); // 通知地址 $this->notify_url自己定義
$unifiedOrder->setParameter("trade_type","JSAPI"); // 交易類型
return $unifiedOrder->getPrepayId();
}
~~~
## 獲取JsApi$getParameters參數
接下來封裝獲取Jsapi的參數
~~~
/**
* 獲取JsApi$getParameters參數
* #User: Mikkle
* #Email:776329498@qq.com
* #Date:
* @param string $unified_order
* @return string
*/
protected function getParameters($unified_order=''){
$jsApi= new JaApi();
$jsApi->setPrepayId($unified_order);
$jsApiParameters = $jsApi->getParameters();
return $jsApiParameters;
}
~~~
## 準備工作都OK的 下面進入主菜
## 根據訂單號支付訂單的方法
### 首先根據訂單號查找訂單
~~~
$param=$this->order_table_param;
$order_info=Db::table($param['table'])
->field(' '. $param['no_field'].' , '.$param['amount_field'].' ')
->where($param['map'][0])
->where($param['state_field'],'=','0')
->where(['order_no'=>$order_no])
->find();
~~~
我是為了通用性 搞了一個訂單查詢這么復雜 你如果就一個程序用 重新寫一些也可以
### 判斷訂單狀態
~~~
if (!$order_info) return ['code'=>1010,'msg'=>'訂單不存在或者已經是完成狀態'];
$data=[
'order_no'=>$order_no,
'amount'=>$order_info['amount'],
];
~~~
如何訂單不存在或者已經完成返回錯誤信息
如何訂單存在 獲取訂單金額
### 進行統一下單 獲取統一下單的訂單號
~~~
$unified_order = $this->unifiedOrder($data); //統一下單
$this->unified_order=$unified_order;
~~~
### 獲取JsApi$getParameters參數
~~~
$jsApiParameters=$this->getParameters($unified_order);
~~~
## 最后整合代碼整合一個方法
整合好的代碼如下
~~~
/**
* 根據訂單號支付訂單返回$getParameters參數
* #User: Mikkle
* #Email:776329498@qq.com
* #Date:
* @param string $order_no
* @param array $param
* @return array
*/
public function payByOrderNo($order_no='2017020453102495',$param=[]){
$param=$this->order_table_param;
$order_info=Db::table($param['table'])
->field(' '. $param['no_field'].' , '.$param['amount_field'].' ')
->where($param['map'][0])
->where($param['state_field'],'=','0')
->where(['order_no'=>$order_no])
->find();
if (!$order_info) return ['code'=>1010,'msg'=>'訂單不存在或者已經是完成狀態'];
$data=[
'order_no'=>$order_no,
'amount'=>$order_info['amount'],
];
$unified_order = $this->unifiedOrder($data); //統一下單
$this->unified_order=$unified_order;
$jsApiParameters=$this->getParameters($unified_order);
return ['code'=>1001,'order_no'=>$order_no,'jsApiParameters'=>$jsApiParameters,'amount'=>$order_info['amount'],];
}
~~~
## 最后 在你的項目的控制器中調用這個方法就能支付了
~~~
<?php
/**
* Created by PhpStorm.
* User: Mikkle
* Email:776329498@qq.com
* Date: 2017/2/8
* Time: 11:32
*/
namespace app\api\controller;
class WxpayAction extends Auth
{
public function index($order_no='2017020453102495'){
$data=controller('base/WxPay')->payByOrderNo($order_no);
$this->assign('amount',$data['amount']);
$this->assign('order_no',$order_no);
$this->assign("jsApiParameters" ,$data['jsApiParameters']);
$this->assign('openid',$this->open_id);
return $this->fetch('wxpay/pay');
}
}
~~~
模版的源代碼
~~~
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width,initial-scale=1,user-scalable=0">
<title>微信安全支付</title>
<link rel="stylesheet" href="__WECHAT_CSS__/weui.css"/>
<link rel="stylesheet" href="__WECHAT_EXAMPLE__/example.css"/>
<script type="text/javascript">
function okAjax(out_trade_no,ext){
$.ajax({
type:"post",
url:"",
data:{out_trade_no:out_trade_no,ext:ext},
dataType:"text",
success:function(data){
if(data){
wx.closeWindow()
}
},
async:false
});
}
//調用微信JS api 支付
function jsApiCall()
{
WeixinJSBridge.invoke(
'getBrandWCPayRequest',
{$jsApiParameters},
function(res){
if(res.err_msg == "get_brand_wcpay_request:ok" ) {
okAjax('{$order_no}','{$order_no}');
}else{
WeixinJSBridge.log(res.err_msg);
// alert('{$order_no}');
alert('支付失敗!'+res.err_code+res.err_desc+res.err_msg);
}
}
);
}
function callpay()
{
if (typeof WeixinJSBridge == "undefined"){
if( document.addEventListener ){
document.addEventListener('WeixinJSBridgeReady', jsApiCall, false);
}else if (document.attachEvent){
document.attachEvent('WeixinJSBridgeReady', jsApiCall);
document.attachEvent('onWeixinJSBridgeReady', jsApiCall);
}
}else{
jsApiCall();
}
}
</script>
</head>
<body ontouchstart>
<div class="weui_msg">
<div class="weui_icon_area"><i class="weui_icon_success weui_icon_msg"></i></div>
<div class="weui_text_area">
<h2 class="weui_msg_title"><font color="#888">你將為{$order_no}支付{$amount}元</font></h2>
</div>
</div>
<div class="weui_opr_area">
<p class="weui_btn_area">
<button type="button" class="weui_btn weui_btn_primary" onclick="callpay()" >立即支付</button>
<a href="javascript:;" onclick="wx.closeWindow()" class="weui_btn weui_btn_default">取消</a>
</p>
</div>
</div>
<script src="__WECHAT_EXAMPLE__/zepto.min.js"></script>
<script src="__WECHAT_EXAMPLE__/router.min.js"></script>
<script src="__WECHAT_EXAMPLE__/example.js"></script>
</body>
</html>
~~~
- 序言及更新日志
- 前言一 開發PHP必備的環境(你可以不看)
- LinUX系統ThinkPHP5鏈接MsSQL數據庫的pdo_dblib擴展
- centos7.2掛載硬盤攻略
- Centos系統Redis安裝及Redis的PHP擴展安裝
- Centos系統增加Swap(系統交換區)的方法
- 前言二 開發PHP軟件配置和介紹(你依然可以不看)
- 數據庫SQL文件
- 本地Git(版本控制)的搭建
- GIT遠程倉庫的克隆和推送
- Git常用命令
- PHP面向對象思想實戰經驗領悟
- PHP面向對象實戰----命名空間
- PHP面向對象實戰----繼承
- 基類實戰--底層方法封裝
- 基類實戰--構造函數實戰
- 基類實戰--析構函數的使用
- TP5實戰開發前篇---控制器(controller)
- 控制器中Request類的使用
- 控制器中基類的使用
- TP5實戰開發前篇---模型篇(model)
- TP5實戰開發前篇---驗證器篇(Validate)
- TP5實戰課程入門篇---花拳繡腿
- 模塊以及類的文件的建立
- Api開發------單條信息顯示
- Api開發---單條信息復雜關聯顯示
- Api開發---查詢信息緩存Cache的應用
- TP5實戰技巧---開發思路 引路造橋
- TP5實戰技巧---整合基類 化繁為簡
- TP5實戰課程入門篇---數據操作
- Api開發---數據的添加和修改
- API開發---快速開發API通用接口
- TP5專用微信sdk使用教程
- THINKPHP5微信SDK更新記錄及升級指導
- TP5專用SDK 微信參數配置方法
- 微信公眾號推送接口對接教程
- 微信推送接口對接示例含掃描登錄微信端部分
- TP5專用微信支付SDK使用簡介
- TP5專用支付寶支付SDK使用說明
- 使用NW將開發的網站打包成桌面應用
- TP5高階實戰課程 進階篇概述
- 進階篇一 實戰開發之習慣及要求
- 進階篇二 實戰開發之控制器
- 控制器基類之控制器基類使用方法
- 控制器基類之控制器基類常用方法分享
- 控制器基類之構造函數的使用方法
- 進階篇三 實戰開發之權限控制
- TP5實戰源碼 --- 全局用戶信息驗證類Auth
- TP5實戰源碼 --- 微信Auth實戰開發源碼
- 進階篇四 實戰開發之模型
- 模型基類之模型基類的用途
- 模型基類之常用數據處理方法
- 模型邏輯層之實戰代碼(含事務)
- 模型實戰開發之模型常用方法
- 模型實戰源碼 --- 樂觀鎖的應用
- 模型實戰技巧---Model事件功能的使用
- 模型事件實戰應用---數據庫操作日志
- 進階篇五 實戰開發之緩存(Cache)
- TP5實戰源碼---應用緩存獲取城市信息
- TP5實戰源碼---應用緩存獲取分類詳情
- 進階篇六 TP5類庫的封裝和使用
- DataEdit快捷操作類庫
- ShowCode快捷使用類庫
- 阿里大于 短信API接口 TP5專用類庫
- DatabaseUpgrade數據庫對比及更新類庫
- AuthWeb權限類使用說明
- 進階篇七 服務層的應用
- 服務層源碼示例
- 服務層基類源碼
- 進階篇八 應用層Redis數據處理基類
- Redis服務層基類源碼
- 進階篇九 使用Redis類庫處理一般的搶購(秒殺)活動示例
- 進階篇十 某大型項目應用本Redis類源碼示例(含事務 樂觀鎖)
- 進階篇十一 邏輯層的應用
- 邏輯層基類源碼
- 進階篇 服務層代碼示例
- 高階實戰課程 進階篇持續新增中
- 高階篇一 TP5命令行之守護任務源碼
- TP5實戰源碼 --- 命令行
- TP5實戰源碼 --- 通過shell建立PHP守護程序
- 高階篇二 使用Redis隊列發送微信模版消息
- 高階篇二 之 Worker隊列基類源碼
- 高階篇三 TP5實戰之Redis緩存應用
- Redis實戰源碼之Hash專用類庫源碼
- Redis實戰源碼之Model類結合
- Redis實戰源碼之模型Hash基類源碼
- Redis實戰源碼之Hash查詢使用技巧
- Redis實戰源碼之 shell腳本中redis賦值和取值
- 高階篇四 Swoole的實戰應用
- swoole基類代碼
- Swoole擴展WebsocketServer專用類
- 基于Swoole的多Room聊天室的程序
- Swoole守護服務shell源碼
- 高階篇五 命令行異步多進程隊列類的應用
- tp_worker類源碼
- WorkerBase
- WorkerCommand
- WorkerRedis
- Redis類
- CycleWorkBase
- WorkerHookBase異步鉤子
- 隊列日志SQL
- 高階篇六 定時執行隊列類庫以及使用方法
- 定時隊列類庫源碼
- 高階篇七 異步執行循環隊列類庫以及使用教程
- CycleWorkBase源碼
- 高階實戰課程 進階篇持續新增中
- Extend便捷類庫源碼庫
- 阿里相關類庫
- SendSms--驗證碼API接口文件
- 權限相關類庫目錄
- AuthWeb 權限驗證類庫
- Redis便捷操作類庫(20171224更新)
- Redis
- Tools工具類庫集
- Curl類庫
- DataEdit
- Rand類庫
- ShowCode類庫
- Upload類庫
- 附件集合
- 附件一:微信支付 實戰開發源碼
- 微信支付類庫源代碼
- Common_util_pub.php
- DownloadBill_pub.php
- JsApi_pub.php
- NativeCall_pub.php
- NativeLink_pub.php
- OrderQuery_pub.php
- Refund_pub.php
- RefundQuery_pub.php
- SDKRuntimeException.php
- ShortUrl_pub.php
- UnifiedOrder_pub.php
- Wxpay_client_pub.php
- Wxpay_server_pub.php
- WxPayConf_pub.php
- 微信支付回調頁面源碼
- 附件二 順豐快遞BSP接口實戰開發源碼
- 順豐快遞BSP接口實戰開發源碼
- 順豐BSP基類
- 順豐BSP基礎代碼
- 順豐BSP下單接口
- 順豐BSP查單接口
- 順豐BSP確認/取消接口
- 附件三 APP注冊登陸接口源碼(含融云平臺接口)
- 附件四 TP5訂單Model(含事務 獲取器 修改器等方法)
- 附錄五 RSA加密解密
- Rsa文件源碼
- 附件六 阿里大于短信接口
- 附件七 AES加解密類
- AES加解密類源碼
- 附件八 TP5路由設置源碼
- 附件九 TP5 Excel導入導出下載便捷類庫
- Excel類庫TP5源碼
- 附件十 TP5便捷操作Redis類庫源碼
- TP5源碼 Redis操作便捷類庫
- 附件十一 TP5源碼 上傳文件入庫類源碼
- 上傳類Upload源碼
- Upload類上傳配置文件
- 存儲圖像文件的數據庫SQL文件
- 存儲文件的數據庫SQL文件
- 附件十二 TP5 圖片處理增強類 支持縮略圖在線顯示
- 附件十三 微信推送消息接口類庫源碼
- 附件十三 微信推送消息接口類庫源碼 之 基類
- 附件十四 存儲微信昵稱的處理方法