Linux与Unix文件通配符

Fit*_*ell 4 unix linux bash

我希望在大写字母的Linux目录中获取一个文件列表.在Unix中,它很简单

ls [AZ]*

但是在Linux中,我看到的匹配似乎不区分大小写:

=> ls
A.txt  b.txt  B.txt  c.txt  C.txt

=> ls [A]*
A.txt

=> ls [AB]*
A.txt  B.txt

=> ls [ABC]*
A.txt  B.txt  C.txt
Run Code Online (Sandbox Code Playgroud)

 
=> ls [AC]*A.txt b.txt B.txt c.txt C.txt

=> ls [b]*
b.txt
Run Code Online (Sandbox Code Playgroud)

 
=> ls [ac]*A.txt b.txt B.txt c.txt

在Unix端运行相同的命令就像我期望的那样.这是Linux一直表现的方式吗?使用awk解决这个问题很容易,所以我不是那样寻找解决方案,但我想知道我以前是否从未注意到这一点.提前致谢.

Nah*_*eul 7

结果取决于不同的shell选项,特别是:nocasematch nocaseglob以及LC_COLLATE变量(用于sort,[ - ]等)

$ shopt extglob nocasematch nocaseglob
extglob         off
nocasematch     off
nocaseglob      off


$ printf "%s\n" a A b B c C | sort
a
A
b
B
c
C
Run Code Online (Sandbox Code Playgroud)

所以[AC]范围包含b和c,但不是a,[ac]也应该包含A但不包含C.

$ printf "%s\n" a A b B c C | LC_COLLATE=C sort
A
B
C
a
b
c
Run Code Online (Sandbox Code Playgroud)

给出了不同的结果.

$ LC_COLLATE=C ls [A-C]*
Run Code Online (Sandbox Code Playgroud)

应返回预期结果,此语法仅为新进程执行(ls)设置变量,但不在当前shell进程中设置.

编辑:感谢评论,以前的命令是错误的,因为在设置LC_COLLATE之前处理扩展,最简单的是分成两个命令.

$ export LC_COLLATE=C
$ ls [A-C]*
Run Code Online (Sandbox Code Playgroud)

  • 在`LC_COLLATE = C ls [AC]*`的例子中,`LC_COLLATE`被设置用于执行`ls` - 但是在**执行发生之前,glob被扩展**.这也是为什么`s = hello echo $ s`不能像人们期望的那样工作的原因. (2认同)