成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

按下ls -l *.py并回車,Shell都為我們做了什么?

系統
shell 是如何理解和解釋這些命令的?屏幕的背后都做些什么?比如說,當我們執行 ls -l *.py 的時候,shell 都做了哪些事情?了解了這些,可以更好的使用 Unix 類操作系統,今天我們就來一探究竟。

[[437059]]

你是否想過,當你在 shell 上執行一個命令時,unix 的 shell 到底做了哪些事情?shell 是如何理解和解釋這些命令的?屏幕的背后都做些什么?比如說,當我們執行 ls -l *.py 的時候,shell 都做了哪些事情?了解了這些,可以更好的使用 Unix 類操作系統,今天我們就來一探究竟。

0、什么是 shell

shell 通常是一個命令行界面,它將操作系統的服務暴露給人類使用或其他程序。在 shell 啟動后,shell 通常會通過顯示提示來等待用戶的輸入。下圖描述了基本的 UNIX 和 Windows shell 提示。

所以 shell 會提示用戶輸入命令。現在是用戶輸入命令的時候了。那么 shell 是如何獲取用戶輸入的命令并進行解釋的呢?為了理解這一點,讓我們將它們分為 4 個步驟,分別是:

  • 獲取并解析用戶輸入
  • 識別命令及命令的參數
  • 查找命令
  • 執行命令

現在詳細展開:

1、獲取并解析用戶輸入

比如說,在 shell 上輸入了 ls -l *.py 并回車,shell 內部會調用一個叫 getline()「聲明在#include 中,下同」 的函數來讀取用戶輸入的命令,用戶輸入的命令字符串作為標準輸入流,一旦按下回車,表示一行結束,getline() 就會將輸入的字符串存儲到緩沖區中。

  1. ssize_t getline(char **restrict lineptr, size_t *restrict n, FILE *restrict stream); 

函數參數說明:

  • lineptr: 緩沖區
  • n: 緩沖區大小
  • stream: 流,這里就是標準輸入流

現在讓我們看一下代碼:

  1. char *input_buffer; 
  2.  
  3. size_t b_size; 
  4.  
  5. b_size = 32; // size of the buffer 
  6.  
  7. input_buffer = malloc(sizeof(char) * b_size); // the buffer to store the user input 
  8.  
  9. getline(&input_buffer, &b_size, stdin); // gets the line and stores it in input_buffer 

一旦用戶按下回車,就會調用 getline() ,將用戶輸入的字符串或命令將存儲在 input_buffer 中。所以現在 shell 已經獲取了用戶輸入,那么下一步是什么?

2、識別命令及命令的參數

現在 shell 已經知道你輸入了字符串是 'ls -l *.py' 但是,還需要知道這里面哪個是命令,哪個是命令的參數,誰來做這個事情呢?那就是函數 strtok()「#include 」。

strtok() 將一個字符串標記為分隔符,在這個例子中分隔符是一個空格。所以一個空格告訴 strtok() 它是一個詞的結尾。因此 input_buffer 中的第一個標記或單詞是命令 (ls),其余的單詞或標記(-l 和 *.py)是命令的參數。因此,一旦 shell 標記了字符串,它就會將它們存儲在一個變量中,以便以后使用。

  1. char *strtok(char *restrict str, const char *restrict delim); 

參數說明:

  • str: 要標記的字符串
  • delim: 分隔符

函數 strtok() 接受字符串和分隔符作為參數,返回一個指向標記字符串的指針。具體的執行代碼如下所示:

  1. char *input_buffer, *args, *delim_args, *command_argv[50]; 
  2. int i; 
  3.  
  4. i = 0; 
  5. delim_args = " \t\r\n\v\f"; // the delimeters 
  6. args = strtok(input_buffer, delim_args); // stores the token inside args 
  7. while (args) 
  8.  command_argv[i] = args; // stores the token in command_argv 
  9.  args = strtok(NULL, delim_args); 
  10.  i++; 
  11. command_argv[i] = NULL; // sets the last entity of command_argv to NULL 

command_argv[i] = NULL; // sets the last entity of command_argv to NULL

command_argv 保存了命令字符串,其內容如下:

  1. command_argv[0] = "ls" 
  2.  
  3. command_argv[1] = "-l" 
  4.  
  5. command_argv[2] = "*.py" 
  6.  
  7. command_argv[3] = NULL 

好了,command_argv[0] 是命令,其他的都是它的參數,最后一個是 NULL,表示命令的結束。命令字符串已經拆解完畢了,下一步就是查找命令。

3、查找命令

第二步已經知道,用戶要執行的命令就是 ls,那么去哪里查找這個命令呢?shell 回去環境變量 PATH 中去查找,PATH 這個環境變量就是存儲可執行命令的位置的。

不過,一個 PATH 存儲的路徑可不止一個:

如何在這么多路徑中高效的查找到 ls 命令呢?這就需要 access() 「#include 」 函數:

  1. int access(const char *pathname, int mode); 

參數及返回值說明:

  • pathname: 文件/可執行文件的路徑
  • mode: 模式,我們使用 X_OK 來檢查文件是否存在
  • 返回值:如果文件存在,返回 0,否則返回 -1
  1.  char *path_buff, *path_dup, *paths, *path_env_name, *path[50]; 
  2.  int i; 
  3.  
  4.  i = 0; 
  5.  path_env_name = "PATH"
  6.  path_buff = getenv(path_env_name); /* get the variable of PATH environment */ 
  7.  path_dup = _strdup(path_buff); /* this function is found below */ 
  8.  paths = strtok(path_dup, ":"); /* tokenizes it */ 
  9.  while (paths) 
  10.  { 
  11.   path[i] = paths; 
  12.   paths = strtok(NULL":"); 
  13.   i++; 
  14.  } 
  15.  path[i] = NULL; /* terminates it with NULL */ 
  16.  
  17. /** 
  18. * _strdup - duplicates a string 
  19. * @from: the string to be duplicated 
  20. Return: ponter to the duplicated string 
  21. */ 
  22. char *_strdup(char *from
  23.  int i, len; 
  24.  char *dup_str; 
  25.  
  26.  len = _strlen(from) + 1; 
  27.  dup_str = malloc(sizeof(int) * len); 
  28.  i = 0; 
  29.  
  30.  while (*(from + i) != '\0'
  31.  { 
  32.   *(dup_str + i) = *(from + i); 
  33.   i++; 
  34.  } 
  35.  *(dup_str + i) = '\0'
  36.  
  37.  return (dup_str); 

上面代碼中的 path 數組存儲所有 PATH 位置并且以 NULL 終止。因此,可以將每個 PATH 位置與命令連接起來,并使用 access() 函數執行存在性檢查:

  1.  char *command_file, *command_path, *path[50]; 
  2.  int i; 
  3.  
  4.  i = 0; 
  5.  command_path = malloc(sizeof(char) * 50); 
  6.  while (path[i] != NULL
  7.  { 
  8.   _strcat(path[i], command_file, command_path); /* this function is found below */ 
  9.   stat_f = access(command_path, X_OK); /* and checks if it exists */ 
  10.   if (stat_f == 0) 
  11.    return (command_path); /* returns the concatenated string if found */ 
  12.  
  13.   i++; 
  14.  } 
  15.  return NULL; /* otherwise returns NULL */ 
  16.  
  17. /** 
  18. * _strcat - concatenates two strings and saves it to a blank string 
  19. * @path: the path string 
  20. * @command: the command 
  21. * @command_path: the string to store the concatenation 
  22. Return: Always void 
  23. */ 
  24. void _strcat(char *path, char *command, char *command_path) 
  25.  int i, j; 
  26.  
  27.  i = 0; 
  28.  j = 0; 
  29.  
  30.  while (*(path + i) != '\0'
  31.  { 
  32.   *(command_path + i) = *(path + i); 
  33.   i++; 
  34.  } 
  35.  *(command_path + i) = '/'
  36.  i++; 
  37.  
  38.  while (*(command + j) != '\0'
  39.  { 
  40.   *(command_path + i) = *(command + j); 
  41.   i++; 
  42.   j++; 
  43.  } 
  44.  *(command_path + i) = '\0'

一旦找到命令,就會返回命令的完整路徑,否則就返回 NULL,然后 shell 會顯示命令不存在的錯誤。

現在假如命令找到了,然后呢?

4、執行命令

命令一旦找到,就是執行它的時候了,問題是怎么執行呢?

執行命令,需要借助函數 execve()「#include 」中:

  1. int execve(const char *pathname, char *const argv[], 
  2.  
  3. char *const envp[]); 

參數說明:

  • pathname: 可執行文件的完整路徑
  • argv: 命令的參數
  • envp: 環境變量列表

execve() 會執行找到的命令,返回一個整數表示執行結果。

但是現在如果 shell 只是運行 execve(),就會出現問題。execve() 調用后不返回標準輸出的信息,這是不好的,因為用戶需要執行的結果。所以為了解決這個問題,shell 在子進程中執行命令。因此,一旦在子進程內執行完成,父進程就會收到信號并且程序流繼續。所以為了執行命令,shell 使用 fork() 創建了一個子進程。(fork 聲明在#include 中)

  1. pid_t fork(void); 

fork() 通過復制調用進程來創建一個新進程。新進程稱為子進程。調用進程稱為父進程。fork() 在父進程中返回子進程的進程 ID,在子進程中返回 0:

  1.  char *command, *command_argv[50], **env; 
  2.  pid_t child_pid; 
  3.  int status; 
  4.  
  5.  get_each_command_argv(command_argv, input_buffer); /* this function is found below */ 
  6.  child_pid = fork(); 
  7.  if (child_pid == -1) 
  8.   return (0); 
  9.  
  10.  if (child_pid == 0) 
  11.  { 
  12.   if (execve(command, command_argv, env) == -1) 
  13.    return (0); 
  14.  } 
  15.  else 
  16.   wait(&status); 
  17.  
  18. /** 
  19. * get_each_command_argv - stores all the arguments \ 
  20. *             of the input command to the list 
  21. * @command_argv: the command argument list 
  22. * @input_buffer: the input buffer 
  23. Return: Always void 
  24. */ 
  25. void get_each_command_argv(char **command_argv, char *input_buffer) 
  26.  char *args, *delim_args; 
  27.  int i; 
  28.  
  29.  delim_args = " \t\r\n\v\f"
  30.  args = strtok(input_buffer, delim_args); 
  31.  
  32.  i = 0; 
  33.  while (args) 
  34.  { 
  35.   command_argv[i] = args; 
  36.   args = strtok(NULL, delim_args); 
  37.   i++; 
  38.  } 
  39.  command_argv[i] = NULL

shell 使用 wait()(函數聲明在#include ) 在程序流繼續之前等待子進程的狀態變化,并再次為用戶顯示提示。

  1. pid_t wait(int *wstatus); 

wstatus:是一個指向整數的指針,可以用來標識子進程是如何終止的。

shell 在子進程內執行命令,然后 wait() 等待子進程完成。所以這樣用戶就可以得到命令的結果,并且可以在 shell 顯示其提示后輸入另一個命令。

所以最后當子進程完成時顯示 ls -l *.py 的結果,并且由于我們已經等待子進程結束,這意味著給出了命令的結果。所以現在 shell 可以再次顯示它的提示以再次等待用戶輸入。這將繼續循環,除非用戶鍵入 exit。

 

責任編輯:武曉燕 來源: Python七號
相關推薦

2024-09-02 13:59:16

2011-04-19 10:04:25

NeopPIshell網站后門

2020-08-13 08:04:31

配置跨域框架

2010-03-12 14:28:52

三層交換技術

2020-09-01 11:40:01

HTTPJavaTCP

2020-10-09 08:59:55

輸入網址解密

2015-08-04 09:33:24

Coding.net

2016-01-29 10:05:13

酒店服務業大數據大數據分析

2011-03-31 09:20:45

URLDNSWeb應用程序

2020-12-15 06:57:24

java服務器

2022-12-07 07:33:54

Java啟動類項目

2021-05-07 11:25:29

項目網關流量

2014-07-17 15:52:00

Android L

2021-03-15 08:40:42

Vue組件函數

2025-02-21 10:01:35

2022-03-11 21:28:31

部署開發服務器

2011-11-29 09:10:11

Hadoop

2020-09-13 09:22:51

Linuxls命令

2019-12-24 11:19:44

容器DockerLinux

2020-11-16 15:47:05

SaaS軟件轉型
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲大片在线观看 | 日韩在线免费播放 | 亚洲欧美日韩国产综合 | 亚洲欧洲精品成人久久奇米网 | 日韩最新网站 | 蜜桃综合在线 | 国产精品免费一区二区三区四区 | 日韩中文字幕一区二区 | 中文字幕国产第一页 | 国产永久免费 | 久久亚洲欧美日韩精品专区 | 久久成人国产精品 | 成人在线h | 一区二区成人在线 | 中文字幕精品一区二区三区精品 | 亚洲午夜av久久乱码 | 激情久久av一区av二区av三区 | 国产精品3区 | 中文字幕av高清 | 欧美日韩在线一区二区 | 久久精品一级 | 久久久久久久久久久成人 | 91精品国产乱码久久久久久久久 | 日本在线观看视频 | 成人精品一区二区三区 | 欧美一区二区三区一在线观看 | 亚洲欧美国产毛片在线 | 亚洲成av人片在线观看 | 亚洲国产黄色av | 一区二区视频在线 | 成人精品一区二区三区中文字幕 | 亚洲精品国产精品国自产在线 | 日韩欧美在线观看视频 | 国产蜜臀97一区二区三区 | 亚洲精品视频免费观看 | 男女网站免费观看 | 欧美日韩一区在线观看 | 成人在线视频免费观看 | 日韩在线第一 | 国产精品久久av | 先锋av资源在线 |