Linux shell:利用sed批量更改文件名

一、示例

1、去除特定字符

目标:将 2017-01-01.jpg、2018-01-01.jpg 改为 20170101.jpg、20180101.jpg
方法:将所有 – 替换为空

for file in `ls | grep .jpg`
do
    newfile=`echo $file | sed 's/-//g'`
    mv $file $newfile
done

这里使用sed进行标准输出的字符串替换,其通用格式如下:

stdout | sed 's/pattern/replace/'

上述示例中,在末尾添加g用于替换所有匹配项,而不仅仅替换第一个匹配项。

2、中间插入字符

目标:将 book01.txt、paper02.txt 改为 book-01.txt、paper-02.txt

方法:用分组匹配分别获取待插入位置两侧的字符串,再通过反向引用实现替换

for file in `ls | grep .txt`
do
    newfile=`echo $file | sed 's/\([a-z]\+\)\([0-9]\+\)/\1-\2/'`
    mv $file $newfile
done

二、分析

上述示例首先通过 ls 和 grep 命令得到待改名的文件列表,然后用 sed 命令进行字符串的替换,最后再使用 mv 命令来完成文件名的更改。

获取待改名文件列表的方法有很多,可以通过 find 命令,也可以直接给出字符串,我们将在下文中提到。

注意 for 循环后面的 ls | grep .txt,这条命令用两个反单引号括了起来,与 $(ls | grep .txt) 的作用相同,被包围的字符串会被当作命令执行,然后返回字符串结果。

1、文件名包含空格的解决方法

我们可以直接将文件列表写到 for 循环中,而不是通过命令来得到,例如:

for file in "file1 file2 file3"
do
    ...
done

可以看到 for 循环是通过空格来分割字符串,因此如果待更改的文件名中包含空格的话,就会被拆分成多个文件名,从而出错。

要解决这个问题,我们可以将 IFS(内部字段分隔符)设置为换行符 \n,这样一来,for 循环就会按行来获取变量的值,确保每次获取的确实是一个完整的文件名。

设置 IFS 变量的命令需要放在 for 循环之前:

IFS=$'\n'
for file in `ls`
do
    ...
done

也可以直接使用 while read 命令一次读取一行到变量 file 中:

ls | grep "*.txt" | while read file
do
    ...
done

2、使用 find 获取文件列表

之前的示例中,我们是通过 ls 命令来获取文件列表。该命令只能获取某个目录的文件,而且没法进行多种条件的筛选。

而一说到文件的查找,不得不提到功能强大的 find 命令。该命令可以在多个层次的目录中查找文件,并能够设定诸如创建时间、文件大小、所有者等多种条件,查找起文件来特别方便灵活。

用 find 命令来获取文件列表,再用 sed 命令配合正则表达式来修改文件名,这两个命令的结合几乎能完成所有常见的批量改名任务。

例如,将所有大于1M,且后缀为txt或jpg的文件,由形如 book_20170101.txt、image_20170101.jpg 的文件改名为 20170101-book.txt、20170101-image.jpg,代码如下:

for file in `find . -size +1M -name "*_*.txt" -o -name "*_*.jpg"`
do
    newfile=`echo $file | sed 's/\([a-z]\+\)_\([0-9]\+\)./\2-\1./'`
    mv $file $newfile
done
标签:ShellLinux 发布于:2019-11-09 17:05:55