4

请耐心等待,这篇文章会有点长...

我有一堆文件,其中一些具有简单而干净的名称(例如 1E01.txt),而另一些则具有很多附加功能:

Sample2_Name_E01_-co_032.txt
Sample2_Name_E02_-co_035.txt
...
Sample12_Name_E01_-co_061.txt

等等。这里重要的是“Sample”后面的数字和“ Name ”后面的字母+数字——其余的都是一次性的。如果我去掉不重要的部分,文件名将减少为与“干净”文件名(2E01.txt、2E02.txt、...、12E01.txt)相同的模式。我设法用以下表达式重命名了文件(我自己想出了这个,不知道是否非常优雅但工作正常):

rename -v 's/Sample([0-9]+)_Name_([A-Z][0-9]+).*/$1$2\.txt/' *.txt

现在,第二部分是为只有一位数字的文件名添加前导零,例如 1E01.txt 变成 01E01.txt。我已经设法做到了这一点(在另一个 StackExchange 帖子中找到并修改了它):

rename -v 'unless (/^[0-9]{2}.*\.txt/) {s/^([0-9]{1}.*\.txt)$/0$1/;s/0*([0-9]{2}\..*)/$1/}' *.txt

所以我终于想到了我的问题:有没有办法在一个重命名命令中合并两个表达式?我知道我可以做一个 bash 脚本来自动化这个过程,但我想要的是找到一个一次性重命名解决方案。

谢谢

4

2 回答 2

11

您可以尝试使用此命令重命名1-file.txt0001-file.txt

# fill zeros
$ rename 's/\d+/sprintf("%04d",$&)/e' *.txt

您可以稍微更改命令以满足您的需要。

于 2012-11-06T12:13:10.123 回答
1

好吧,如果那是您的“解析”正则表达式,那么您就是在限制脚本可以作用于与该模式匹配的文件。因此,sprintf使用相同的文字字符串不是更特殊的情况,您可以这样做:

s{Sample(\d+)_Name_(\p{IsUpper})(\d+)}
 {sprintf "Sample%02d_Name_%s%03d", $1, $2, $3}e
 ;

在这里,您再次使用相同的已知功能并简单地格式化随附的数字。

  • /e开关用于 'eval',它会为每个匹配评估替换为 Perl。
  • 我将您的一些表达式重命名为更标准的字符类符号:[A-Z]成为属性类\p{IsUpper}[0-9]成为数字代码\d(也可能\p{IsDigit})。
于 2012-11-06T12:15:22.830 回答