說到shell通配符(wildcard),你們在使用時候會常常用到。下面是一個實例:html
[chengmo@localhost ~/shell]$ ls a.txt b.txt c.old [chengmo@localhost ~/shell]$ ls *.txt a.txt b.txt [chengmo@localhost ~/shell]$ ls d*.txt ls: 沒法訪問 d*.txt: 沒有那個文件或目錄
從上面這個實例,不知道你們有沒有發現問題呢。咱們先了解一下,通配符相關知識,再分析下這個實例吧。linux
通配符是由shell處理的(不是由所涉及到命令語句處理的,其實咱們在shell各個命令中也沒有發現有這些通配符介紹), 它只會出如今 命令的「參數」裏(它不用在 命令名稱裏, 也不用在 操做符上)。當shell在「參數」中遇到了通配符時,shell會將其看成路徑或文件名去在磁盤上搜尋可能的匹配:若符合要求的匹配存在,則進行代換(路徑擴展);不然就將該通配符做爲一個普通字符傳遞給「命令」,而後再由命令進行處理。總之,通配符 實際上就是一種shell實現的路徑擴展功能。在 通配符被處理後, shell會先完成該命令的重組,而後再繼續處理重組後的命令,直至執行該命令。正則表達式
咱們回過頭分析上面命令吧:在第2個命令中,*.txt 實際shell搜索文件,找到了符合條件的文件,命令會變成:ls a.txt b.txt ,實際在執行ls 時候傳給它的是a.txt b.txt .shell
而命令3,d*.txt 因爲當前目錄下面沒有這樣的文件或目錄,直接將」d*.txt」 做爲ls 參數,傳給了 ls .這個時候」*」 只是一個普通的 ls 參數而已,已經失去了它通配意義。 因爲找不到文件,因此會出現:沒法訪問提示!bash
瞭解了shell通配符,咱們如今看下,shell常見通配符有那一些了。spa
shell常見通配符:命令行
字符 | 含義 | 實例 |
* | 匹配 0 或多個字符 | a*b a與b之間能夠有任意長度的任意字符, 也能夠一個也沒有, 如aabcb, axyzb, a012b, ab。 |
? | 匹配任意一個字符 | a?b a與b之間必須也只能有一個字符, 能夠是任意字符, 如aab, abb, acb, a0b。 |
[list] | 匹配 list 中的任意單一字符 | a[xyz]b a與b之間必須也只能有一個字符, 但只能是 x 或 y 或 z, 如: axb, ayb, azb。 |
[!list] | 匹配 除list 中的任意單一字符 | a[!0-9]b a與b之間必須也只能有一個字符, 但不能是阿拉伯數字, 如axb, aab, a-b。 |
[c1-c2] | 匹配 c1-c2 中的任意單一字符 如:[0-9] [a-z] | a[0-9]b 0與9之間必須也只能有一個字符 如a0b, a1b... a9b。 |
{string1,string2,...} | 匹配 sring1 或 string2 (或更多)其一字符串 | a{abc,xyz,123}b a與b之間只能是abc或xyz或123這三個字符串之一。 |
須要說明的是:通配符看起來有點象正則表達式語句,可是它與正則表達式不一樣的,不能相互混淆。把通配符理解爲shell 特殊代號字符就可。並且涉及的只有,*,? [] ,{} 這幾種。code
shell 除了有通配符以外,由shell 負責預先先解析後,將處理結果傳給命令行以外,shell還有一系列本身的其餘特殊字符。orm
字符 | 說明 |
IFS | 由 <space> 或 <tab> 或 <enter> 三者之一組成(咱們經常使用 space )。 |
CR | 由 <enter> 產生。 |
= | 設定變量。 |
$ | 做變量或運算替換(請不要與 shell prompt 搞混了)。 |
> | 重導向 stdout。 * |
< | 重導向 stdin。 * |
| | 命令管線。 * |
& | 重導向 file descriptor ,或將命令置於背境執行。 * |
( ) | 將其內的命令置於 nested subshell 執行,或用於運算或命令替換。 * |
{ } | 將其內的命令置於 non-named function 中執行,或用在變量替換的界定範圍。 |
; | 在前一個命令結束時,而忽略其返回值,繼續執行下一個命令。 * |
&& | 在前一個命令結束時,若返回值爲 true,繼續執行下一個命令。 * |
|| | 在前一個命令結束時,若返回值爲 false,繼續執行下一個命令。 * |
! | 執行 history 列表中的命令。* |
加入」*」 都是做用在命令名直接。能夠看到shell 元字符,基本是做用在命令上面,用做多命令分割(或者參數分割)。所以看到與通配符有相同的字符,可是實際上做用範圍不一樣。因此不會出現混淆。htm
如下是man bash 獲得的英文解析:
metacharacter
A character that, when unquoted, separates words. One of the following:
| & ; ( ) < > space tab
control operator
A token that performs a control function. It is one of the following symbols:
|| & && ; ;; ( ) | <newline>
有時候,咱們想讓 通配符,或者元字符 變成普通字符,不須要使用它。那麼這裏咱們就須要用到轉義符了。 shell提供轉義符有三種。
字符 | 說明 |
‘’(單引號) | 又叫硬轉義,其內部全部的shell 元字符、通配符都會被關掉。注意,硬轉義中不容許出現’(單引號)。 |
「」(雙引號) | 又叫軟轉義,其內部只容許出現特定的shell 元字符:$用於參數代換 `用於命令代替 |
\(反斜槓) | 又叫轉義,去除其後緊跟的元字符或通配符的特殊意義。 |
man bash 英文解釋以下:
There are three quoting mechanisms: the escape character, single quotes, and double quotes.
實例:
[chengmo@localhost ~/shell]$ ls \*.txt ls: 沒法訪問 *.txt: 沒有那個文件或目錄 [chengmo@localhost ~/shell]$ ls '*.txt' ls: 沒法訪問 *.txt: 沒有那個文件或目錄 [chengmo@localhost ~/shell]$ ls 'a.txt' a.txt [chengmo@localhost ~/shell]$ ls *.txt a.txt b.txt
能夠看到,加入了轉義符 「*」已經失去了通配符意義了。
看到上面說的這些,想必你們會問到這個問題是,有這麼想特殊字符,通配符,那麼 shell在獲得一條命令,到達是怎麼樣處理的呢?咱們看下下面的圖:
若是用雙引號包括起來,shell檢測跳過了1-4步和9-10步,單引號包括起來,shell檢測就會跳過了1-10步。也就是說,雙引號 只通過參數擴展、命令代換和算術代換就能夠送入執行步驟,而單引號轉義符直接會被送入執行步驟。並且,不管是雙引號轉義符仍是單引號轉義符在執行的時候可以告訴各個命令自身內部是一體的,可是其自己在執行時是並非命令中文本的一部分。
——EOF——
Refer:
[1] 命令行通配符教程
http://www.ruanyifeng.com/blog/2018/09/bash-wildcards.html
[2] Linux Shell 通配符、元字符、轉義符使用實例介紹
http://www.cnblogs.com/chengmo/archive/2010/10/17/1853344.html