偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

使用 Python 創(chuàng)建你自己的 Shell (上)

開發(fā) 開發(fā)工具
我很想知道一個 shell (像 bash,csh 等)內(nèi)部是如何工作的。于是為了滿足自己的好奇心,我使用 Python 實現(xiàn)了一個名為 yosh (Your Own Shell)的 Shell。本文章所介紹的概念也可以應用于其他編程語言。

[[169212]]

我很想知道一個 shell (像 bash,csh 等)內(nèi)部是如何工作的。于是為了滿足自己的好奇心,我使用 Python 實現(xiàn)了一個名為yosh (Your Own Shell)的 Shell。本文章所介紹的概念也可以應用于其他編程語言。

(提示:你可以在這里查找本博文使用的源代碼,代碼以 MIT 許可證發(fā)布。在 Mac OS X 10.11.5 上,我使用 Python 2.7.10 和 3.4.3 進行了測試。它應該可以運行在其他類 Unix 環(huán)境,比如 Linux 和 Windows 上的 Cygwin。)

讓我們開始吧。

步驟 0:項目結構

對于此項目,我使用了以下的項目結構。 

  1. yosh_project 
  2. |-- yosh 
  3.  |-- __init__.py 
  4.  |-- shell.py 

yosh_project 為項目根目錄(你也可以把它簡單命名為 yosh)。

yosh 為包目錄,且 __init__.py 可以使它成為與包的目錄名字相同的包(如果你不用 Python 編寫的話,可以忽略它。)

shell.py 是我們主要的腳本文件。

步驟 1:Shell 循環(huán)

當啟動一個 shell,它會顯示一個命令提示符并等待你的命令輸入。在接收了輸入的命令并執(zhí)行它之后(稍后文章會進行詳細解釋),你的 shell 會重新回到這里,并循環(huán)等待下一條指令。

在 shell.py 中,我們會以一個簡單的 main 函數(shù)開始,該函數(shù)調(diào)用了 shell_loop() 函數(shù),如下: 

  1. def shell_loop(): 
  2.  # Start the loop here 
  3. def main(): 
  4.  shell_loop() 
  5. if __name__ == "__main__"
  6.  main() 

接著,在 shell_loop() 中,為了指示循環(huán)是否繼續(xù)或停止,我們使用了一個狀態(tài)標志。在循環(huán)的開始,我們的 shell 將顯示一個命令提示符,并等待讀取命令輸入。 

  1. import sys 
  2. SHELL_STATUS_RUN = 1 
  3. SHELL_STATUS_STOP = 0 
  4. def shell_loop(): 
  5.     status = SHELL_STATUS_RUN 
  6.     while status == SHELL_STATUS_RUN: 
  7.         ### 顯示命令提示符 
  8.         sys.stdout.write('> '
  9.         sys.stdout.flush() 
  10.         ### 讀取命令輸入 
  11.         cmd = sys.stdin.readline() 

之后,我們切分命令tokenize輸入并進行執(zhí)行execute(我們即將實現(xiàn) tokenize 和 execute 函數(shù))。

因此,我們的 shell_loop() 會是如下這樣: 

  1. import sys 
  2. SHELL_STATUS_RUN = 1 
  3. SHELL_STATUS_STOP = 0 
  4. def shell_loop(): 
  5.     status = SHELL_STATUS_RUN 
  6.     while status == SHELL_STATUS_RUN: 
  7.         ### 顯示命令提示符 
  8.         sys.stdout.write('> '
  9.         sys.stdout.flush() 
  10.         ### 讀取命令輸入 
  11.         cmd = sys.stdin.readline() 
  12.         ### 切分命令輸入 
  13.         cmd_tokens = tokenize(cmd) 
  14.         ### 執(zhí)行該命令并獲取新的狀態(tài) 
  15.         status = execute(cmd_tokens) 

這就是我們整個 shell 循環(huán)。如果我們使用 python shell.py 啟動我們的 shell,它會顯示命令提示符。然而如果我們輸入命令并按回車,它會拋出錯誤,因為我們還沒定義 tokenize 函數(shù)。

為了退出 shell,可以嘗試輸入 ctrl-c。稍后我將解釋如何以優(yōu)雅的形式退出 shell。

步驟 2:命令切分tokenize

當用戶在我們的 shell 中輸入命令并按下回車鍵,該命令將會是一個包含命令名稱及其參數(shù)的長字符串。因此,我們必須切分該字符串(分割一個字符串為多個元組)。

咋一看似乎很簡單。我們或許可以使用 cmd.split(),以空格分割輸入。它對類似 ls -a my_folder 的命令起作用,因為它能夠?qū)⒚罘指顬橐粋€列表 ['ls', '-a', 'my_folder'],這樣我們便能輕易處理它們了。

然而,也有一些類似 echo "Hello World" 或 echo 'Hello World' 以單引號或雙引號引用參數(shù)的情況。如果我們使用 cmd.spilt,我們將會得到一個存有 3 個標記的列表 ['echo', '"Hello', 'World"'] 而不是 2 個標記的列表 ['echo', 'Hello World']。

幸運的是,Python 提供了一個名為 shlex 的庫,它能夠幫助我們?nèi)缒Хò愕胤指蠲睢?提示:我們也可以使用正則表達式,但它不是本文的重點。) 

  1. import sys 
  2. import shlex 
  3. ... 
  4. def tokenize(string): 
  5.     return shlex.split(string) 
  6. ... 

然后我們將這些元組發(fā)送到執(zhí)行進程。

步驟 3:執(zhí)行

這是 shell 中核心而有趣的一部分。當 shell 執(zhí)行 mkdir test_dir 時,到底發(fā)生了什么?(提示: mkdir 是一個帶有test_dir 參數(shù)的執(zhí)行程序,用于創(chuàng)建一個名為 test_dir 的目錄。)

execvp 是這一步的首先需要的函數(shù)。在我們解釋 execvp 所做的事之前,讓我們看看它的實際效果。 

  1. import os 
  2. ... 
  3. def execute(cmd_tokens): 
  4.     ### 執(zhí)行命令 
  5.     os.execvp(cmd_tokens[0], cmd_tokens) 
  6.     ### 返回狀態(tài)以告知在 shell_loop 中等待下一個命令 
  7.     return SHELL_STATUS_RUN 
  8. ... 

再次嘗試運行我們的 shell,并輸入 mkdir test_dir 命令,接著按下回車鍵。

在我們敲下回車鍵之后,問題是我們的 shell 會直接退出而不是等待下一個命令。然而,目錄正確地創(chuàng)建了。

因此,execvp 實際上做了什么?

execvp 是系統(tǒng)調(diào)用 exec 的一個變體。***個參數(shù)是程序名字。v 表示第二個參數(shù)是一個程序參數(shù)列表(參數(shù)數(shù)量可變)。p 表示將會使用環(huán)境變量 PATH 搜索給定的程序名字。在我們上一次的嘗試中,它將會基于我們的 PATH 環(huán)境變量查找mkdir 程序。

(還有其他 exec 變體,比如 execv、execvpe、execl、execlp、execlpe;你可以 google 它們獲取更多的信息。)

exec 會用即將運行的新進程替換調(diào)用進程的當前內(nèi)存。在我們的例子中,我們的 shell 進程內(nèi)存會被替換為 mkdir 程序。接著,mkdir 成為主進程并創(chuàng)建 test_dir 目錄。***該進程退出。

這里的重點在于我們的 shell 進程已經(jīng)被 mkdir 進程所替換。這就是我們的 shell 消失且不會等待下一條命令的原因。

因此,我們需要其他的系統(tǒng)調(diào)用來解決問題:fork。

fork 會分配新的內(nèi)存并拷貝當前進程到一個新的進程。我們稱這個新的進程為子進程,調(diào)用者進程為父進程。然后,子進程內(nèi)存會被替換為被執(zhí)行的程序。因此,我們的 shell,也就是父進程,可以免受內(nèi)存替換的危險。

讓我們看看修改的代碼。 

  1. ... 
  2. def execute(cmd_tokens): 
  3.     ### 分叉一個子 shell 進程 
  4.     ### 如果當前進程是子進程,其 `pid` 被設置為 `0` 
  5.     ### 否則當前進程是父進程的話,`pid` 的值 
  6.     ### 是其子進程的進程 ID。 
  7.     pid = os.fork() 
  8.     if pid == 0: 
  9.     ### 子進程 
  10.         ### 用被 exec 調(diào)用的程序替換該子進程 
  11.         os.execvp(cmd_tokens[0], cmd_tokens) 
  12.     elif pid > 0: 
  13.     ### 父進程 
  14.         while True
  15.             ### 等待其子進程的響應狀態(tài)(以進程 ID 來查找) 
  16.             wpid, status = os.waitpid(pid, 0) 
  17.             ### 當其子進程正常退出時 
  18.             ### 或者其被信號中斷時,結束等待狀態(tài) 
  19.             if os.WIFEXITED(status) or os.WIFSIGNALED(status): 
  20.                 break 
  21.     ### 返回狀態(tài)以告知在 shell_loop 中等待下一個命令 
  22.     return SHELL_STATUS_RUN 
  23. ... 

當我們的父進程調(diào)用 os.fork() 時,你可以想象所有的源代碼被拷貝到了新的子進程。此時此刻,父進程和子進程看到的是相同的代碼,且并行運行著。

如果運行的代碼屬于子進程,pid 將為 0。否則,如果運行的代碼屬于父進程,pid 將會是子進程的進程 id。

當 os.execvp 在子進程中被調(diào)用時,你可以想象子進程的所有源代碼被替換為正被調(diào)用程序的代碼。然而父進程的代碼不會被改變。

當父進程完成等待子進程退出或終止時,它會返回一個狀態(tài),指示繼續(xù) shell 循環(huán)。

運行

現(xiàn)在,你可以嘗試運行我們的 shell 并輸入 mkdir test_dir2。它應該可以正確執(zhí)行。我們的主 shell 進程仍然存在并等待下一條命令。嘗試執(zhí)行 ls,你可以看到已創(chuàng)建的目錄。

但是,這里仍有一些問題。

***,嘗試執(zhí)行 cd test_dir2,接著執(zhí)行 ls。它應該會進入到一個空的 test_dir2 目錄。然而,你將會看到目錄并沒有變?yōu)? test_dir2。

第二,我們?nèi)匀粵]有辦法優(yōu)雅地退出我們的 shell。

 

責任編輯:龐桂玉 來源: Linux中國
相關推薦

2021-02-05 15:50:27

PythonShell命令

2017-02-09 09:30:18

UbuntuDokuWikiApache

2011-07-07 10:39:10

yum源createrepo

2018-10-19 09:50:15

Linuxman手冊Linux命令

2014-03-06 09:23:19

Git服務器Github

2020-04-02 18:30:28

PythonGitHub編程語言

2020-04-02 16:02:44

PythonGithub博客

2018-05-21 14:44:33

LinuxshellPython

2014-07-30 09:35:36

DockerPaaS

2023-07-30 17:34:53

KV存儲ChunkPosit

2017-03-07 17:12:46

LinuxUbuntu發(fā)行版

2018-07-27 16:18:30

PythonTwitter機器人

2024-01-18 07:53:37

2018-03-22 11:00:45

PythonRSS

2021-05-26 10:21:31

Python音樂軟件包

2009-12-11 18:00:50

Linux KVM

2022-05-17 12:45:31

LinuxLinux發(fā)行版

2009-05-25 09:48:43

2011-05-17 10:46:14

TAP

2021-01-21 16:03:15

Java文本編輯器編程語言
點贊
收藏

51CTO技術棧公眾號