假设我有一个包含一些文件的目录:
$ ls
a.c b.c e.c k.cpp s.java
如何在没有文件扩展名的情况下显示结果(点后面的部分,包括那个点)?喜欢这个:
$ <some command>
a
b
e
k
s
使用 sed?
ls -1 | sed -e 's/..*$//'
ls | while read fname
do
echo ${fname%%.*}
done
试试看。
ls -a | cut -d "." -f 1
人 (1) 切
非常方便,-d 开关定义了分隔符和您想要的字段的 -f。
编辑:包括河流的场景也是小菜一碟,因为切割也可以从最后开始,尽管逻辑有些不同。这里有一堆随机名称的文件的例子,有些有两个点,有些只有一个点,有些没有扩展名:
runlevel0@ubuntu:~/test$ ls
test.001.rpx test.003.rpx test.005.rpx test.007.rpx test.009.rpx testxxx
test.002.rpx test.004.rpx test.006.rpx test.008.rpx test_nonum test_xxx.rtv
runlevel0@ubuntu:~/test$ ls | cut -d "." -f -2
test.001
test.002
test.003
test.004
test.005
test.006
test.007
test.008
test.009
test_nonum
testxxx
test_xxx.rtv
在字段编号之前使用减号可以消除所有指示的字段(在本例中为 1,2),并将其放在后面使其从末尾开始计数。
除字段外,此表示法可用于偏移量和字符(请参阅手册页)
如果您已经知道文件的扩展名,则可以在手册页中使用 basename
:
基名 - 从文件名中删除目录和后缀
不幸的是,如果您尝试过滤单个扩展,它最有用,在您的情况下,命令是:
basename -s .c -a $(ls *.c) && basename -s .cpp -a $(ls *.cpp) && basename -s .java -a $(ls *.java)
输出:
a
b
e
k
s
for f in *; do printf "%sn" ${f%%.*}; done
为什么有效?
${string%%substring}
从$string后面删除最长的$substring匹配项。
例如,这将处理mypackage.pkg.tar.xz
--> mypackage
。
相比之下:
${string%substring} 从$string后面删除$substring的最短匹配项。
也就是说${string%substring}
只会删除最终的扩展名,即 mypackage.pkg.tar.xz
--> mypackage.pkg.tar
附带说明一下,优先使用 printf
echo
.语法稍微复杂一些,但它适用于更广泛的系统。
如果您只想查看文件,而不是目录:
for f in *; do if [[ -f ${f} ]]; then printf "%sn" ${f%%.*}; fi; done