当Linux下文件名中出现空格这类特殊情况话,如何查找或确认那些文件名后有空格呢? 又怎么批量替换处理掉这些空格呢?
方法1:
输入文件名后使用Tab键,如果使用Tab键后面出现/ / /这样的可见字符,那么该文件名包含空格。当然,这个方法弊端很大,例如,效率低下,不能批量查找,只有当你怀疑某个文件名后有空格,这个方法才比较凑效。另外,不能查找文件中间包含空格的文件名。如下测试所示:
[root@DB-Server kerry]# cat >"test.txt "it is only for test![1]+ Stopped cat > "test.txt "[root@DB-Server kerry]# cat >"tes t.txt"it is only for test too![2]+ Stopped cat > "tes t.txt"[root@DB-Server kerry]# ls test.txtls: test.txt: No such file or directory[root@DB-Server kerry]# ls testtest~ test1.py test.py test.sh test.txt [root@DB-Server kerry]# ls test.txt/ / / / test.txt [root@DB-Server kerry]# ls testest~ test1.py test.py test.sh tes t.txt test.txt
方法2:
使用find命令查找文件名中包含空格的文件。
[root@DB-Server kerry]# find . -type f -name "* *" -print./test.txt ./tes t.txt
那么如何将这些空格替换掉呢? 下面脚本可以替换文件中间的空格,用下划线替换空格,但是只能替换文件中间的空格,并不能替换文件名后面的空格。如下测试所示:
find . -type f -name "* *" -print |while read name; dona=$(echo $name | tr ' ' '_')if [[ $name != $na ]]; thenmv "$name" "$na"fidone
上面脚本只能将文件名中间有空格的替换为下划线。那么如何解决文件名后有空格的情况呢? 可以用其它shell脚本实现,如下所示:
[root@DB-Server kerry]# rm -rf *[root@DB-Server kerry]# cat >"test.txt "12[root@DB-Server kerry]# cat >"tes t.txt"12[root@DB-Server kerry]# find . -type f -name "* *" -print./test.txt ./tes t.txt[root@DB-Server kerry]# for file in *; do mv "$file" `echo $file | tr ' ' '_'` ; done[root@DB-Server kerry]# find . -type f -name "* *" -print[root@DB-Server kerry]# ls -lrttotal 8-rw-r--r-- 1 root root 0 Nov 13 10:04 test.txt-rw-r--r-- 1 root root 0 Nov 13 10:04 tes_t.txt
如上所示,虽然文件名中间的空格被替换为了下划线,但是后面的空格没有替换为下划线,而是将那些空格直接截断了。Why?下面使用sed命令也是如此
[root@DB-Server kerry]# rm -rf *[root@DB-Server kerry]# cat >"test.txt "12[root@DB-Server kerry]# cat >"tes t.txt"12[root@DB-Server kerry]# find . -type f -name "* *" -print./test.txt ./tes t.txt[root@DB-Server kerry]# for i in *' '*; do mv "$i" `echo $i | sed -e 's/ /_/g'`; done[root@DB-Server kerry]# find . -type f -name "* *" -print[root@DB-Server kerry]# ls -lrttotal 8-rw-r--r-- 1 root root 0 Nov 13 09:29 test.txt-rw-r--r-- 1 root root 0 Nov 13 09:29 tes_t.txt[root@DB-Server kerry]# [root@DB-Server kerry]#
其实,这个是因为读取文件名是$file 与"$file"是不同的,$file不会识别文件名后面的空格,而"$file"才会失败文件名后面的空格。所以上面脚本其实只是取巧而已。
[root@DB-Server kerry]# rm -rf *;[root@DB-Server kerry]# cat >"test.txt "123[root@DB-Server kerry]# for file in *; do echo "$file"; echo "$file" | wc -m ; done;test.txt 13[root@DB-Server kerry]# for file in *; do echo $file; echo $file | wc -m ; done;test.txt9[root@DB-Server kerry]#
所以,正确的替换空格的命令应该为如下:
方案1:
[root@DB-Server kerry]# rm -rf *[root@DB-Server kerry]# cat >"test.txt "123456[root@DB-Server kerry]# find . -type f -name "* *" -print./test.txt [root@DB-Server kerry]# for file in *; do mv "$file" `echo "$file" | tr ' ' '/n'` ; done[root@DB-Server kerry]# find . -type f -name "* *" -print[root@DB-Server kerry]# ls test.txttest.txt[root@DB-Server kerry]#
方案2:
[root@DB-Server kerry]# [root@DB-Server kerry]# rm -rf *[root@DB-Server kerry]# cat >"test.txt "123456[root@DB-Server kerry]# for file in *' '*; do mv "$file" `echo "$file" | sed -e 's/ /n/g'`; done[root@DB-Server kerry]# find . -type f -name "* *" -print
但是对于文件名中间包含空格的情况,上面两个脚本都无法完美解决。如下所示:
[root@DB-Server kerry]# [root@DB-Server kerry]# rm -rf *[root@DB-Server kerry]# cat >"tes t.txt"123456[root@DB-Server kerry]# for file in *; do mv "$file" `echo "$file" | tr ' ' '_'` ; done[root@DB-Server kerry]# find . -type f -name "* *" -print[root@DB-Server kerry]# ls -lrt total 8-rw-r--r-- 1 root root 7 Nov 13 16:00 tes_t.txt[root@DB-Server kerry]# [root@DB-Server kerry]# rm -rf *[root@DB-Server kerry]# cat >"tes t.txt"123456[root@DB-Server kerry]# cat >"test.txt "654321[root@DB-Server kerry]# find . -type f -name "* *" -print./test.txt ./tes t.txt[root@DB-Server kerry]# for file in *; do mv "$file" `echo "$file" | tr ' ' '_'` ; done[root@DB-Server kerry]# find . -type f -name "* *" -print[root@DB-Server kerry]# ls -lrttotal 12-rw-r--r-- 1 root root 0 Nov 13 15:59 tes_t.txt-rw-r--r-- 1 root root 7 Nov 13 15:59 test.txt____
当然对于这两种特殊情况,上面脚本都不能一起处理,如上所示,后面的空格会被替换成了下划线。这反而不是我们想要的,反而最上面的那两种脚本,可以误打误撞的解决这两种问题。当然让前提是你得知其然知其所以然!
新闻热点
疑难解答