大家好,我是小小明,今天我要給大家分享一個(gè)用python實(shí)現(xiàn)的仿Linux的tree命令。
詳見:https:///project/filestools/
通過以下命令安裝即可直接使用:
pip install filestools -U
安裝后的使用示例:

對如何編碼實(shí)現(xiàn)感興趣的童鞋可以繼續(xù)往下看。
首先看看Linux下的tree命令效果如何:
Linux與Windows的tree命令
Linux的tree命令演示
在CentOS的Linux系統(tǒng)下,我們可以再使用yum命令安裝tree之后使用tree。安裝命令:
yum install tree
然后使用tree命令的樹形顯示效果:
[root@iZwz9afmmytm54pshbwmebZ 018]# tree
.
├── css
│ ├── abstract\ blue\ lights\ orange\ bokeh\ gaussian\ blur\ 1920x1200\ wallpaper_www.wallpaperhi.com_43.jpg
│ ├── default.css
│ ├── font
│ │ ├── DS-DIGIB.TTF
│ │ ├── DS-DIGII.TTF
│ │ ├── DS-DIGI.TTF
│ │ └── DS-DIGIT.TTF
│ ├── jquery-ui.css
│ ├── mobile.css
│ ├── normalize.css
│ └── style.css
├── images
│ ├── btn01slider2.png
│ ├── charts.png
│ ├── logofont.png
│ ├── logoline1.png
│ ├── logoline2.png
│ ├── logoline3.png
│ └── logoline.png
├── index.html
├── js
│ ├── common.js
│ ├── index.js
│ ├── jquery-1.8.3.min.js
│ └── jquery_and_jqueryui.js
└── less
└── style.less
加上-C參數(shù)對各種類型加上不同的顏色:

加上-s參數(shù)能額外列出文件或目錄的大小,-h參數(shù)用于自動(dòng)修正顯示單位:

tree命令參數(shù)說明:
- -a 顯示所有文件和目錄。
- -A 使用ASNI繪圖字符顯示樹狀圖而非以ASCII字符組合。
- -C 在文件和目錄清單加上色彩,便于區(qū)分各種類型。
- -d 顯示目錄名稱而非內(nèi)容。
- -D 列出文件或目錄的更改時(shí)間。
- -f 在每個(gè)文件或目錄之前,顯示完整的相對路徑名稱。
- -F 在執(zhí)行文件,目錄,Socket,符號連接,管道名稱名稱,各自加上"*","/","=","@","|"號。
- -g 列出文件或目錄的所屬群組名稱,沒有對應(yīng)的名稱時(shí),則顯示群組識別碼。
- -i 不以階梯狀列出文件或目錄名稱。
- -L level 限制目錄顯示層級。
- -l 如遇到性質(zhì)為符號連接的目錄,直接列出該連接所指向的原始目錄。
- -n 不在文件和目錄清單加上色彩。
- -N 直接列出文件和目錄名稱,包括控制字符。
- -p 列出權(quán)限標(biāo)示。
- -P<范本樣式> 只顯示符合范本樣式的文件或目錄名稱。
- -q 用"?"號取代控制字符,列出文件和目錄名稱。
- -s 列出文件或目錄大小。
- -t 用文件和目錄的更改時(shí)間排序。
- -u 列出文件或目錄的擁有者名稱,沒有對應(yīng)的名稱時(shí),則顯示用戶識別碼。
- -x 將范圍局限在現(xiàn)行的文件系統(tǒng)中,若指定目錄下的某些子目錄,其存放于另一個(gè)文件系統(tǒng)上,則將該子目錄予以排除在尋找范圍外。
Windows的tree命令
Windows上也有tree命令,只不過沒有Linux平臺的tree命令強(qiáng)大。
我們看看顯示效果:
D:\QMDownload\source\test>tree /F
文件夾 PATH 列表
卷序列號為 5A3F-F8A8
D:.
│ index.html
│
├─css
│ │ abstract blue lights orange bokeh gaussian blur 1920x1200 wallpaper_www.wallpaperhi.com_43.jpg
│ │ default.css
│ │ jquery-ui.css
│ │ mobile.css
│ │ normalize.css
│ │ style.css
│ │
│ └─font
│ DS-DIGI.TTF
│ DS-DIGIB.TTF
│ DS-DIGII.TTF
│ DS-DIGIT.TTF
│
├─images
│ btn01slider2.png
│ charts.png
│ logofont.png
│ logoline.png
│ logoline1.png
│ logoline2.png
│ logoline3.png
│
├─js
│ common.js
│ index.js
│ jquery-1.8.3.min.js
│ jquery_and_jqueryui.js
│
└─less
style.less
個(gè)人只能說勉強(qiáng)還行吧,也能看清楚樹形結(jié)構(gòu)。
Python自制tree命令
下面呢,我們考慮使用Python來仿制這樣的命令。
os模塊基礎(chǔ)代碼
關(guān)于Python打印樹形目錄結(jié)構(gòu),我已經(jīng)在4年前使用Java寫過一個(gè)不夠完善的代碼。19年國慶學(xué)python的時(shí)候才用os模塊重寫了一下:
import os
def show_dir(path, layer=0):
listdir = os.listdir(path)
for i, file in enumerate(listdir, 1):
file_path = os.path.join(path, file)
print("│ " * (layer - 1), end="")
if (layer > 0):
print("└─" if i == len(listdir) else "├─", end="")
print(file)
if (os.path.isdir(file_path)):
show_dir(file_path, layer + 1)
show_dir("test")
結(jié)果:
css
├─abstract blue lights orange bokeh gaussian blur 1920x1200 wallpaper_www.wallpaperhi.com_43.jpg
├─default.css
├─font
│ ├─DS-DIGI.TTF
│ ├─DS-DIGIB.TTF
│ ├─DS-DIGII.TTF
│ └─DS-DIGIT.TTF
├─jquery-ui.css
├─mobile.css
├─normalize.css
└─style.css
images
├─btn01slider2.png
├─charts.png
├─logofont.png
├─logoline.png
├─logoline1.png
├─logoline2.png
└─logoline3.png
index.html
js
├─common.js
├─index.js
├─jquery-1.8.3.min.js
└─jquery_and_jqueryui.js
less
└─style.less
還行,但是對于很深的目錄缺點(diǎn)也很明顯,例如出現(xiàn)這樣的顯示情況:

Rich庫的使用示例
要自行完全實(shí)現(xiàn)Linux的樹形目錄比較復(fù)雜,所幸python有個(gè)第三方庫rich
中的Tree模塊能支持彩色和樹形輸出。
安裝命令:
pip install rich
詳細(xì)使用方式可以參考官方文檔:https://rich./en/stable/
Tree模塊的使用示例:https://github.com/willmcgugan/rich/blob/master/examples/tree.py
這個(gè)官方的代碼示例就是專門用來樹形顯示目錄的,我們可以復(fù)制粘貼到j(luò)upyter中稍微改改玩一下。
上述代碼底部修改的部分:
directory = os.path.abspath("test")
tree = Tree(
f":open_file_folder: [link file://{directory}]{directory}",
guide_style="bold bright_blue",
)
walk_directory(pathlib.Path(directory), tree)
print(tree)

顯示效果比Linux的tree命令更秀。不過這個(gè)腳本兼容性較差,Windows控制臺并不支持顯示圖標(biāo)之類的,導(dǎo)致會出現(xiàn)亂碼:

調(diào)用Tree模塊實(shí)現(xiàn)仿Linux樹形顯示目錄效果
由于官方自帶案例秀過頭了兼容性不太好,所以我們自行編碼:
"""
小小明的代碼
CSDN主頁:https://blog.csdn.net/as604049322
"""
__author__ = '小小明'
import os
import sys
import rich
from rich.text import Text
from rich.tree import Tree
def get_file_size(file):
size = os.path.getsize(file)
if size == 0:
return "空文件"
num = 0
while size > 1024:
size /= 1024
num += 1
unit = ["", "KB", "MB", "GB", "TB"]
return f"{size:.2f}".rstrip(".0") + unit[num]
def show_dir(path, tree=None):
if tree is None:
tree = Tree(f"[bold magenta]{os.path.abspath(path)}")
for file in os.listdir(path):
file_path = os.path.join(path, file)
if (os.path.isdir(file_path)):
parent = tree.add(f"[bold magenta]{file}")
show_dir(file_path, parent)
else:
text_filename = Text(file, "green")
text_filename.highlight_regex(r"\.[^.]+$", "bold red")
text_filename.append(f" ({get_file_size(file_path)})", "bold blue")
tree.add(text_filename)
return tree
if __name__ == '__main__':
rich.print(show_dir(sys.argv[1]))
將以上代碼保存為tree.py
,然后在jupyter中執(zhí)行:
from tree import show_dir
import rich
rich.print(show_dir("test"))

在Windows控制臺中的執(zhí)行結(jié)果:
python tree.py test

將腳本上傳到Linux看下Linux下的執(zhí)行效果:

可以看到我們自行編寫的腳本已經(jīng)能夠同時(shí)適用于windows和Linux平臺。
這就是tree模塊核心邏輯的開發(fā),至此我們的目標(biāo)就已經(jīng)達(dá)成。