Linux shell利用sed如何批量更改文件名详解
前言 本文主要给大家介绍了关于Linux shell用sed批量更改文件名的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧。 示例 去除特定字符 目标:将 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用于替换所有匹配项,而不仅仅替换第一个匹配项。 中间插入字符 目标:将 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 循环后面的 文件名包含空格的解决方法 我们可以直接将文件列表写到 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 使用 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 有关 find 命令的更多用法,可以参考 https://www.oudahe.com/p/37077/ 总结 以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对编程小技巧的支持。 (编辑:甘孜站长网) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |