前言

我们在使用 shell 执行任务的过程中,常常会遇到需要处理一批数据的情况,如果我们一个一个的传递参数就会非常的麻烦,这时候就需要用到 shell 的通配符功能了。例如rm *.txt可以删除当前目录下所有的 txt 文件。

功能

shell 通配符起到的是拓展参数的功能,注意 shell 通配符是由 shell 处理的,而不是用到参数的命令或者语句处理的。

例如对于 rm *.txt,shell 在参数遇到通配符的时候,会把这个通配符当做路径或者文件的匹配模式去磁盘上搜索所有的匹配项。

如果存在匹配,则把所有的匹配项替换到参数去,例如上面的命令最终的形式可能是 rm a.txt b.txtrm命令拿到的是实际的文件列表,而不是*.txt

如果不存在匹配或者无法识别该模式,则shell会将该通配符作为一个普通字符传递给命令,然后再由命令去处理。例如如果我们目录下没有任何 txt 文件,执行上面的命令就会报错:no matches found: *.txt

跨目录匹配
通配符只能匹配单层目录,如果要跨目录匹配,则要这样子写:

rm */*.txt

注意,上面的写法只能匹配一级子目录下的 txt 文件,没有办法匹配当前目录以及二级子目录下的文件

如果要匹配当前目录和一级子目录下的txt文件,则要用到多个通配符组合,例如如下命令

ls *{\/*,}.txt

注意,以上命令在当前目录或者一级目录之一没有txt文件的时候,也会报错。

通配符

shell 通配符看起来很像正则表达式,然而并不是正则表达式,它的功能比正则表达式要弱,只支持下面几种通配符形式。

*

匹配 0 或多个字符

?

匹配任意一个字符

[]

匹配 [] 中的任意单一字符,例如[abc]匹配a、b、c中的任何一个字符。[]支持范围匹配,例如 [a-z]匹配所有小写字母。

{,}

匹配{}中被,分隔的任意一个子字符串。例如{AA,BB,CC}.txt匹配到 AA.txtBB.txtCC.txt{}也支持范围匹配,例如{A..Z}匹配所有大写字母

{}和其他通配符不同的地方在于,即使没有匹配到数据,{}依然会展开。例如

# 如果我们目录下没有文件A,B,下面的命令会报错:no matches found: [AB]
echo [A-B]
# 下面的命令则会输出:A B
echo {A-B}

{}支持嵌套,因此可以组合成复杂的模式。例如

echo {a{a..c},b{b..c}}
#输出aa ab ac bb bc

[!]和[^]

匹配除了 [] 中的其他所有字符,也即不匹配[]中的所有字符。

Enjoy it !

版权声明

转载请注明作者和文章出处
作者: X先生
https://segmentfault.com/a/1190000023398532

X先生
317 声望786 粉丝

腾讯TEG研发管理部小小后台攻城狮一枚,负责腾讯敏捷产品研发平台TAPD的基础功能的开发和维护,热爱技术,喜欢分享,欢迎与我交流~