# [教程] 掌握命令行的表格輸出
> ThinkPHP`V5.1.24`引入了一個新的`Table`對象,用于在命令行下面動態輸出表格,剛好最新的`route:list`指令中包含了大部分表格功能的實現,所以是學習Table對象的最好示例,本篇就通過分析該指令的代碼來掌握`Table`對象的用法。
[TOC=2,2]
## 初見端倪
`route:list`指令是一個用于查看路由定義的指令,我們先運行下看下在命令行下輸出的表格效果,有個感性的認識。
隨便在你的路由定義文件中定義一些路由規則,例如下面定義了一個閉包路由、一個資源路由和一個普通路由。
~~~
Route::get('think', function () {
return 'hello,ThinkPHP5!';
});
Route::resource('blog', 'Blog');
Route::get('hello/:name', 'index/hello');
~~~
在命令行下面(切換到項目根目錄)執行:
~~~
php think route:list
~~~
你會看到下面的表格輸出:

> 通過這個指令你可以清晰看到每個資源路由實際上會注冊7個路由規則。
是不是覺得很酷?? 當然了這個指令的功能遠不止這么簡單,我們后面會陸續講解。
## 指令尋址
為了分析這個指令,我們先找到`route:list`指令對應的類文件(指令尋址),打開`\think\Console`類,找到`defaultCommands`屬性

這個屬性定義了核心內置的所有指令,數組的索引就是指令名,鍵值就代表了該指令對應的完整類名。可以看到`route:list`指令對應的就是`think\console\command\RouteList`類,接下來就來給大家分析下這個指令的代碼實現。
## 指令定義
任何一個合法的指令類都必須繼承系統的`think\console\Command`類,`RouteList`類一共130行左右代碼,所以基本上很容易看明白,首先我們看第一個`configure`方法,該方法的作用就是配置當前指令的用法。

`setName`方法就是指定當前類的指令名(注意指令名盡量使用小寫,并且不要包含除`:`、`_`之外的特殊字符)。
`addArgument`方法是添加指令參數,這里添加的style參數表示表格輸出的樣式。該方法的參數可以在`think\console\input\Argument`類的構造方法中查看。

第二個參數表示參數的類型必須或者可選,對于`route:list`指令而言,這個參數是可選的,因此傳入`Argument::OPTIONAL`,并且設置了該參賽的默認值是`default`。
`addOption`方法是添加指令選項,這里添加了`sort`和`more`兩個選項,并且分別設置了簡寫。
具體用法則可以在`think\console\input\Option`類的構造方法中看到。

`setDescription`方法用于指定當前指令的描述信息。
我們可以用`-h`選項來查看下指令配置的最終效果(可以對任何指令使用`-h`或者`--help`選項來查看指令幫助)
~~~
php think route:list -h
~~~

## 撥云見日
第二個方法是`execute`方法,該方法會在指令在執行的時候自動調用。

可以看到`route:list`指令的執行操作很簡單,獲取路由定義列表的內容(這個時候會同時在命令行輸出)后就寫入一個`route_list.php`文件中。所有的真相都在`getRouteList()`這個方法里面了,這也是本文的關鍵所在。
再來深入分析`getRouteList()`方法。

這行代碼可能很多人不明白,我先解釋下。由于ThinkPHP`5.1`的路由采用延時解析機制,如果不開啟測試模式的話,可能沒有辦法獲取到所有的路由規則的,尤其是當使用了路由分組和資源路由的時候,實際上都是在路由匹配后才會真正解析分組或者資源路由下面的實際路由規則。而`setTestMode(true)`的作用就是讓路由規則實時解析,這樣我們才能獲取到所有的路由規則。

這段代碼的作用則是獲取所有的路由定義文件,如果采用了配置數組方式定義的話也會批量導入。
>[danger] 配置數組方式定義路由規則在`5.1`版本中已經不再建議,在下一個大版本中就會廢除,所以盡量不要使用數組方式定義,而改為方法定義

這段代碼的作用是在開啟注解路由的情況下進行注解路由的解析,有興趣的朋友可以通過跟蹤`think\Build`類的`buildRoute`方法的實現深入了解下注解路由是如何實現的,由于不再文本的講解范疇,所以暫且不表。
## 表格輸出
前面鋪墊了那么多,接下來就要進入正題了(原來講了這么多廢話??)

這段代碼首先實例化了一個`think\console\Table`類,并通過`setHeader`方法設置輸出的表頭信息。
`setHeader`方法的參數可以參考`Table`類的代碼

第一個參數是表頭信息(數組),其實還有第二個參數可以控制表頭信息的對齊方式,默認是左對齊,如果要讓表頭信息居中對齊或者居右,可以使用下面的方式
把
~~~
$table->setHeader($header);
~~~
改為居中對齊
~~~
// 表頭居中對齊
$table->setHeader($header, Table::ALIGN_CENTER);
~~~
再次運行指令后,輸出的效果變為:

或者居右對齊
~~~
// 表頭居右對齊
$table->setHeader($header, Table::ALIGN_RIGHT);
~~~

這里判斷是否傳入了`more`選項(參考前面講到的指令配置部分),如果有則附加額外的表頭信息。也就是說,如果使用了`--more`或者`-m`你就能看到關于路由的詳細信息(增加了路由參數和變量規則)。


這段代碼的作用是獲取路由定義列表數據,事實上,`think\Route`類已經封裝好了一個獲取已經定義的路由規則列表方法`getRuleList()`,所以在指令中我們不需要自己實現。有興趣的朋友,可以參考下`think\route\RuleName`類的`setRule()`方法和`getRuleList()`兩個方法的代碼實現,在此略過不提。
由于`getRouteList()`方法獲取的數據是一個三維數組,我們需要重新進行封裝為一個Table類可以輸出的二維數組格式。同樣在封裝單元格數據的時候也判斷是否使用了`--more`選項而使用不同的數據。最終封裝到的二維數組保存到`rows`數組變量中。

這段代碼的作用如果傳入了`--sort`選項,則對二維數組進行指定排序排序。
排序的指定方式有兩種:一種是直接從0開始的序號,0表示對第一列排序,1表示對第二列排序,以此類推,另外一種方式是對指定的列名進行排序。
例如我們要對第三列(請求類型)進行排序,可以用

由于這樣不夠直觀,系統還提供了第二種方式


這段代碼的作用就是給`Table`類實例對象設置表格數據,和`setHeader()`方法一樣,`setRows()`方法也支持指定對齊方式。

如果我們把
~~~
$table->setRows($rows);
~~~
改成:
~~~
// 表格數據居右對齊
$table->setRows($rows, Table::ALIGN_RIGHT);
~~~
再次運行指令后的輸出效果如圖:


這段代碼的作用是使用`setStyle()`方法指定表格輸出的樣式,我們之前看到的都是默認的樣式輸出。
現在我們來看下可以支持哪些樣式,



通過查看Table類的源碼,我們可以看到支持的樣式可以包括:
樣式名|描述
---|---
default|默認樣式
compact|沒有任何表格邊框輸出
markdown|MARKDOWN格式輸出表格
borderless|沒有左右邊框的表格輸出
box|閉合邊框輸出表格
box-double|雙線閉合邊框輸出表格
> 如果沒有指定表格樣式或者指定了一個不存在的樣式,則默認采用`default`樣式風格。
>
最后一步是渲染輸出表格,這是調用的是`Command`類封裝好的`table`方法,當然你也可以自己參考該方法實現。

下面的代碼是`Command`類的`table`方法的實現代碼。

該方法在命令行輸出表格之后返回了渲染的表格內容,是為了便于后面寫入路由定義列表的內容到`route_list.php`文件,可以隨時查看。
有時候需要實現一個跨行跨列的輸出,例如實現一個跨列的單元格,我們在
~~~
$table->setRows($rows);
~~~
后面添加如下代碼:
~~~
// 跨列數據直接傳入字符串
$table->addRow('Extra Route List');
$table->addRow(['test/:name', 'index/test', '*', 'test', '']);
~~~
現在運行指令后輸出效果如圖:

> 目前Table尚不支持跨部分列的表格輸出
## 總結
總結下來,`Table`類的用法其實很簡單
~~~
// 實例化一個Table對象
$table = new Table();
// 設置表頭(可選)
$table->setHeader();
// 設置表格數據
$table->setRows();
// 添加單行數據(可選)
$table->addRow();
// 設置表格樣式(可選)
$table->setStyle();
// 渲染表格輸出
$content = $table->render();
~~~
其實不僅僅是用于命令行輸出,你還可以進行文件寫入的時候生成表格數據內容。
- 值得升級到5.1的18個理由
- 5.1.7版本新特性
- JSON字段類型在ORM中的使用
- 文件下載響應對象
- 教你使用5.1的數組對象查詢
- 模型三大利器之一:搜索器
- 在ThinkPHP中使用Yaconf
- 掌握命令行的表格輸出
- 5.1.25查詢參數綁定的改進
- ThinkPHP安全規范指引
- 巧用數據集的排序功能實現統計排序
- think-orm ——基于5.1的獨立ORM庫
- think-template——基于ThinkPHP的獨立模板引擎
- ThinkPHP5.1.26版本發布——修正版本,包含安全更新
- ThinkPHP5.0和3.2再發安全更新
- 官宣:ThinkPHP發布首個LTS版本
- 你真的了解Db類和模型的正確使用姿勢么?
- 如何更有效的記錄和管理日志
- 模型三大利器之二:修改器
- ThinkPHP5.1.28版本發布——修正上一版本問題,改進關聯查詢
- 模型三大利器之三:獲取器
- API版本控制的幾種思路
- ThinkPHP5.2第一個Beta版本發布測試
- 讓你少犯錯的數據查詢基本原則
- ThinkPHP發布5.1.29版本——常規更新
- 這15個好習慣讓你更容易升級到5.2
- 如何有效提高ThinkPHP的應用性能
- 讓你提高開發效率的查詢技巧
- 模型關聯查詢不完全指南
- 5.2發布Beta2版本——統一和精簡大量用法
- ThinkPHP發布5.1.30版本——支持微秒時間字段寫入
- ThinkPHP的數據緩存使用
- ThinkPHP5.2安裝及入口文件
- ThinkPHP榮獲2018 年度最受歡迎中國開源開發框架第1名
- 5.1路由使用心得技巧
- ThinkPHP5.*版本發布安全更新
- ThinkPHP項目及代碼規范指北
- 5.2版本的設計規范指導
- ThinkPHP5.1.32版本發布——圣誕快樂
- 利用Trait特性給模型增加樂觀鎖功能
- 5.2數據庫和模型的變化(摘要)
- ThinkPHP模板引擎實現和常見問題
- ThinkPHP5.0.24版本發布——安全更新
- 不忘初心,方得始終——ThinkPHP十三周年報告
- ThinkPHP5+相關資源匯總
- 異步社區ThinkPHP周年慶專享優惠活動
- 5.2路由的調整和改進
- ThinkPHP發布5.1.33版本——包含安全更新
- ThinkPHP擴展開發指南
- ThinkPHP發布5.2Beta3版本
- ThinkPHP發布5.1.34版本——喜迎新年
- ThinkPHP發布5.2RC1版本
- ThinkPHP發布5.1.35版本——常規更新
- 5.2配置類的調整
- 5.2時間查詢的改進和優化
- 5.2RC版本升級不完全指導(僅供學習參考)
- ThinkPHP5.2版本正式變更為6.0版本
- ThinkPHP百度云云虛擬主機專享免費活動
- 事件系統以及查詢事件、模型事件的使用
- ThinkPHP6.0RC2版本發布——架構升級、精簡核心
- ThinkPHP5.1.36LTS版本發布——常規更新
- 新版Session和Cookie設計變化
- ThinkPHP5.1.37版本發布——常規更新
- ThinkPHP6.0RC3版本發布——細節完善,體驗優化
- 6.0中間件使用詳解
- Composer各大廠商鏡像地址
- ThinkPHP6.0發布計劃公告
- 「ThinkPHP開發者周刊」招募志愿者
- ThinkPHP6.0日志變化
- ThinkPHP5.1.38版本發布——常規更新
- ThinkPHP6.0RC4版本發布——ORM獨立,日志多通道支持
- ThinkORM2.0開發指南上線
- ThinkPHP6.0RC5版本發布——多應用模式獨立,中間件機制調整
- ThinkPHP6.0版本發布——程序員節福利
- ThinkPHP5.1.39LTS版本發布——常規更新
- ThinkPHP6.0.1版本發布——圣誕快樂!
- 回顧2019,展望2020!
- ThinkPHPV6.0.2版本發布——2020新春快樂!
- 周年福利系列:Swoole合作優惠
- 億速云成為ThinkPHPV6.0獨家贊助發布商??
- 新冠疫情工具和限免資源專題(保持更新中)
- 周年福利系列:創宇信用認證合作優惠
- 周年福利系列:碼云企業版限時10%優惠
- 周年福利系列:想天短說抵現優惠
- think-swoole直播:從零開始掌握swoole開發
- 周年福利系列:B2C開源電商ShopXO授權8折優惠
- 周年福利系列:LayuiAdmin 永久授權限時優惠
- ThinkPHP資源導航站上線——構建生態 服務未來
- ThinkPHP官方技術支持服務和應用服務市場上線公測
- ThinkPHP市場精選——推廣基本要素
- ThinkPHP市場精選——客服聊天專題
- ThinkPHPV6.0.3版本發布——端午安康
- ThinkPHP開發者扶持計劃
- 6.0.3版本關鍵更新及升級事項
- 「ThinkPHP開發者周刊」改版重啟
- ThinkPHP市場精選——企業建站專題
- ThinkPHP 提供統一API接口服務
- ThinkPHP市場精選——直播電商專題
- ThinkAPI服務SDK發布
- 官方服務市場啟用獨立子域名
- ThinkPHP市場精選——刷臉支付專題
- ThinkAPI推出會員服務計劃
- ThinkPHPV6.0.4版本發布——中秋國慶雙節快樂
- ThinkPHPV5.1.40版本發布——常規更新
- 1024程序員節福利走一波
- ThinkPHP V6.0.5版本發布——兼容Composer2.0
- 知識圖譜應用場景——源論技術沙龍
- ThinkPHP5.*版本改進Composer2.0的兼容
- 官方市場雙十一精選推薦
- 技術人做產品有機會么(文末送課程)
- 本周秒殺——古德云售后獲客營銷系統
- ThinkAPI服務更新——支持接口分組和PHP版本依賴調整
- PHP8新特性盤點
- PHP8新特性系列:構造器屬性提升使用及注意事項
- ThinkPHP2021新年寄語
- ThinkPHP V6.0.6&V5.1.41版本發布——兼容PHP8.0
- PHP如何更優雅地調用API接口
- ThinkPHP V6.0.7發布——修正版本
- ThinkAPI服務更新——IP白名單
- 最新版ThinkORM對于時間字段的調整
- ThinkAPI短信接口正式上線
- ThinkPHP V6.0.8版本發布——多環境變量配置支持
- 頂想云寫作服務開啟第一次公測
- ThinkSSL上線——官方SSL/TLS證書服務
- MDBootstrap國內用戶福利——ThinkPHP官方市場首發
- ThinkPHP V6.0.9版本發布——常規更新
- ThinkORM功能盤點——虛擬模型
- 全面支持主流GIT版本庫——云寫作服務第二次公測
- 云寫作服務私有化部署方案之:版本庫私有化
- 看云雙十一活動
- ThinkPHP V6.0.10LTS發布——兼容PHP8.1
- ThinkPHP V6.0.12發布——命令行兼容8.1
- 頂想云知識管理上線公測——構建企業文檔中心和知識庫
- 頂想云上線——助力生態數字化建設
- 618活動進行中——官方市場迎來一波更新
- 頂想云知識管理正式上線——看云文檔啟動遷移服務
- ThinkPHP V6.0.13發布——常規更新
- 頂想云網站助理服務上線——構建產品支持服務
- ThinkPHP發布6.1.0&6.0.14版本——安全更新
- ThinkPHP新版社區上線試運營
- ThinkAPI上架人臉核身接口——助力網站實名認證
- 辭舊迎新——舊版社區停止注冊及發帖
- ThinkPHP6.1.2版本發布——兼容PHP8.2