當你打開終端並輸入命令時會發生什麼?(下)

来源:https://www.cnblogs.com/edisonfish/archive/2023/12/15/17904050.html
-Advertisement-
Play Games

當你打開終端並輸入命令時會發生什麼?(下) 哈嘍大家好,我是鹹魚 我們先來大致回顧一下文章《當你打開終端並輸入命令時會發生什麼?(上)》的內容 終端設備是由電傳打字機演變過來的,電傳打字機通過物理線與大型電腦連接在一塊來實現輸入輸出 如上圖,分別是二戰時期的電傳打字機和西門子 “Fernschei ...


當你打開終端並輸入命令時會發生什麼?(下)

哈嘍大家好,我是鹹魚

我們先來大致回顧一下文章《當你打開終端並輸入命令時會發生什麼?(上)》的內容

終端設備是由電傳打字機演變過來的,電傳打字機通過物理線與大型電腦連接在一塊來實現輸入輸出
image
image
如上圖,分別是二戰時期的電傳打字機和西門子 “Fernscheiber 100” 電傳打字機

隨著技術的不斷發展(尤其是顯示技術),帶顯示屏的終端設備隨之誕生
image
而現在隨著個人電腦的普及,出現了基於屏幕顯示的圖形用戶界面(GUI),演變成了現在的電腦終端

現在的終端大多都是電腦上的一個應用程式,它們通常被稱為終端模擬器,充當用戶與操作系統交互的界面(比如說 Linux 中的 Xterm、Xshell,Windows 中的控制台),而不必使用專門的終端。輸出系統是屏幕,輸入系統是鍵盤

以 Linux 為例,當我們打開終端時,通常會啟動一個 shell 進程,用於與用戶交互。用戶在終端中輸入的命令將傳遞給 shell 進程,然後由 shell 解釋和執行這些命令
image

這個過程包括將用戶輸入的命令解析為操作系統可以理解的指令,執行這些指令,並將執行結果返回給終端顯示給用戶

輸入命令

當我們在終端中輸入命令時,鍵盤輸入的字元會被轉換成相應的字元編碼(比如說 backspace 鍵被轉譯成 ASCII 字元 0x08

這些字元通過終端寫入到 PTY leader,接著 TTY driver 從 PTY leader 中讀取字元並存儲到 line discipline 中(line discipline 為 PTY 兩端之間的中間緩衝區)

不但如此,line discipline 還負責解釋來自 PTY leader 的字元然後根據自己的規則去處理它們(比如進行回退、刪除字元等,或者處理特殊字元)
image
舉個例子,line discipline 收到 backspace 時,它會根據自己的規則解釋成 ERASE 字元,然後進行編輯,方法是刪除最後一個字元

接著將刪除操作返回給 PTY leader,這樣終端就可以從 PTY leader 那裡讀取到更改並將其反映在終端顯示中

需要註意的是,上面這段過程里字元還沒有被寫入到 PTY follower 中,只是處在【編輯】部分

當我們在鍵盤敲下 CTRL+C 時,line discipline 會解釋成 INTR (INTERRUPT 的縮寫),這時候就會向 PTY follower 發送一個 SIGINT 信號去中斷在前臺運行的任何進程

如果不是特殊字元(比如輸入 ls),line discipline 會將字元返回給 PTY leader,終端程式讀取並顯示在屏幕上,這就是為什麼你在鍵盤敲一個字元,顯示器就會顯示一個字元(echo 功能)

現在 shell 進程也會緩衝用戶的輸入,以實現一些高級的功能:比如命令歷史記錄或 tab 鍵自動補全

執行並解析命令

當我們輸入完命令之後,就要按下回車鍵(Enter)來執行命令了

一旦按下回車鍵,line discipline 解釋為換行字元(newline),通常表示為 NL

然後一併將用戶的命令轉發到 PTY follower ,而 shell 進程跟 follower 相連,shell 拿到命令之後就會去解析並執行

當 shell 進程接收到用戶的輸入和換行符時,它會開始解析並執行命令。這個過程包括命令解析、查找可執行文件或內置命令,以及執行相應的操作

首先對命令解析成一個一個 token 併進行語法/語義分析,以 ls 命令為例:

  • ls > foo.txt :正確
  • ls > :語法不正確, > 後面缺少內容
  • ls | foo.txt :語義不正確,管道的兩端都需要是可運行的進程

然後接著解析那些不是 shell 關鍵字或者路徑的 token,shell 需要知道這些 token 的含義,所以 shell 會去根據下麵幾個部分去遞歸查找 token 引用的內容:

  • aliases:命令別名,通常用於縮寫複雜的命令(例如 alias ll="ls -lh"
  • function:函數
  • environment variables:環境變數
  • builtins:shell 內嵌命令(例如 cd pwd exit kill
  • PATH executables:shell 可以找到(通過 $PATH 變數)並運行的外部命令

我們可以通過 type 命令知道對應的類型

[root@minion1 ~]# type ll
ll 是 'ls -l --color=auto' 的別名

[root@minion1 ~]# type cd
cd 是 shell 內嵌

[root@minion1 ~]# type python
python 是 /usr/bin/python

與 shell 內嵌命令不同的是,可執行命令是單獨的程式或腳本文件,具有執行許可權,可以作為單獨的進程執行

當用戶在 shell 中輸入一個命令時,shell 會查找可執行文件的位置,併在找到後創建一個新的子進程來運行該可執行文件,並將相應的命令參數傳遞給這個新的子進程

我們可以通過 pstree 命令來查看進程之間的關係

返回輸出

當 shell 執行完命令之後,把生成的輸出寫入到 PTY follower ,接著傳到 line discipline 中,line discipline 不會處理這些輸出,而是轉發給 PTY leader,然後終端就會讀取並顯示到屏幕上
image
參考文章:https://www.warp.dev/blog/what-happens-when-you-open-a-terminal-and-enter-ls


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Qt 是一個跨平臺C++圖形界面開發庫,利用Qt可以快速開發跨平臺窗體應用程式,在Qt中我們可以通過拖拽的方式將不同組件放到指定的位置,實現圖形化開發極大的方便了開發效率,本章將重點介紹`Slider`滑塊條組件的常用方法及靈活運用。當涉及到C++ Qt開發中的`Slider`滑塊條組件時,你可能會... ...
  • 題目:輸入兩個正整數 m 和 n,求其最大公約數和最小公倍數。 求出最大公約數就行,最小公倍數用m*n除以最大公約數就行 package myself; import java.util.Scanner; /** * @Auther QY * @Date 2023/12/11 */ public c ...
  • 但在微服務架構中,每個微服務通常有多個實例,每個實例具有不同的位置,而且實例會動態變化,比如在負載發生變化時服務會進行擴容或縮容,或者某個實例所在的VM/Container故障後發生遷移,都會導致服務實例地址的變化。因此使用微服務架構開發的應用,必須通過服務註冊和發現技術解決此問題。 1. 簡單概述 ...
  • 1.1系統架構的演變 隨著互聯網的發展,網站應用的規模不斷擴大,常規的應用架構已無法應對,分散式服務架構以及微服務架構勢在必行,亟需一個治理系統確保架構有條不紊的演進。 1.1.1單體應用架構 Web應用程式發展的早期,大部分web工程(包含前端頁面,web層代碼,service層代碼,dao層代碼 ...
  • Quartz.NET 是一個用於在 .NET 應用程式中實現作業調度和定時任務的開源框架。它允許你在應用程式中定義和調度作業,支持複雜的調度需求,例如定時、重覆、錯過執行、依賴性等。下麵,我將通過一個簡單的實例來詳細描述 Quartz.NET 的功能、使用方法,並提供源代碼。 在這個示例中,我將使用 ...
  • 一:背景 1. 講故事 前幾天有位朋友找到我,說他的機器記憶體在不斷的上漲,但在任務管理器中查不出是哪個進程吃的記憶體,特別奇怪,截圖如下: 在我的分析旅程中都是用戶態模式的記憶體泄漏,像上圖中的異常徵兆已經明確告訴你了,不是用戶態程式吃的記憶體,那就是內核態程式吃的,比如: 某些驅動程式 操作系統 從概率 ...
  • Entity Framework Core 使用語言集成查詢 (LINQ) 來查詢資料庫中的數據。 通過 LINQ 可使用 C#(或你選擇的其他 .NET 語言)編寫強類型查詢。 它使用你派生得到的上下文和實體類來引用資料庫對象。 EF Core 將 LINQ 查詢的表示形式傳遞給資料庫提供程式。 ...
  • ZXing.NET 是一個開源的、功能強大的二維碼處理庫,它能夠對二維碼進行解碼(讀取信息)和編碼(生成二維碼)。ZXing 是 "Zebra Crossing" 的縮寫,是一個跨平臺的、用於解碼和生成條形碼和二維碼的庫。以下是一些 ZXing.Net 的主要功能通過實例講解。 1. 生成二維碼 u ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...