我在客户的目录上使用它来重命名文件,每个单词的第一个字母根据他们的要求大写:
rename 's/\b(\w)/\u$1/g' *
这很好用,但它也将扩展名的第一个字母大写,所以我用它来解决这个问题:
rename 's/\.Txt$/.txt/' *.Txt
如果文件夹中的大多数文件具有相同的扩展名(通常是正确的),这可以正常工作,但如果它们非常混合,那就很痛苦了。
为了解决这个问题,我创建了一个看起来像这样的小脚本(不要笑!):
#!/bin/bash
rename 's/\.Txt$/.txt/' *.Txt
rename 's/\.Doc$/.doc/' *.Doc
rename 's/\.Docx$/.docx/' *.Docx
...
rename 's/\.Xlsx$/.xlsx/' *.Xlsx
我忽略了“无法重命名 *.Txt *.txt:没有这样的文件或目录”错误,如果我发现缺少的扩展名,我只需将该行添加到我的脚本中。
我认为这无关紧要,但这些文件位于 Windows 服务器文件共享上,但我正在使用 Ubuntu 16.04LTS 访问它。如果它确实重要,那么我可以先将它们复制到我的本地驱动器,在 Ubuntu 中运行命令,然后在需要时将文件移回。
有没有办法修改第一个重命名命令以忽略扩展名并将它们保留为小写?我可以在更高级别的目录中运行新命令,并通过所有子目录递归吗?
实现此目的的一种方法可能是使用Negative Lookbehind仅匹配单词边界 - 单词字符序列,当它前面没有文字句点时,例如给定
然后
假设您还想避免将复数形式大写
s
,我们可能会将其修改为甚至
这将大写每个单词的第一个字母,除了最后的扩展名:
我用这些文件进行了测试:
它会将它们重命名为:
诀窍是首先将每个第一个字母大写,忽略扩展名,然后返回并使扩展名小写:
s/\b(.+?)\b/\u$1/g;
:这.+?
是一个非贪婪模式,这意味着它会找到最短的匹配。由于它是由单词边界(\b
)锚定的,因此它将找到所有单词(都是因为 finalg
)。然后将它们替换为自身的大写(首字母大写)版本(\l$1
)。s/(.+)\.(.)/$1\.\l$2/
:.+
是贪婪的,所以它会找到最长的匹配。这意味着直到 final 的最长字符串.
,之后将是扩展名(如果有的话)。我们将匹配项替换为扩展名 ($1
)、a.
和第一个字母再次小写的扩展名 ( ) 之前的所有内容\l$2
。递归也很容易。如果你的 shell 是 bash (如果你不知道,它可能是),你可以使用 globstar 选项来
**
匹配 0 个或更多子目录:现在,像这样运行重命名命令(这也适用于当前目录中的任何文件):
要将其限制为仅具有扩展名的文件或目录,请使用
**/*.*
而不是**
.或者,使用
find
:并且,要限制为具有扩展名的文件和目录:
请注意,所有这些解决方案都会愉快地重命名目录和文件。如果您不希望这样,请小心告诉它递归的位置,或者给它一个更具体的模式,例如
*.txt
.在所有示例中,删除 -n 以使它们实际执行某些操作。简单地打印它会做什么而不实际做任何事情的
-n
原因rename
对测试很有用。最明智的方法是处理包括扩展名在内的所有“单词”(使用单词边界
\b
并通过引用$1
任何第一个字符),然后将扩展名本身小写:请注意,这不适用于带有下划线的文件名,即“单词”边界被视为空白(制表符、空格、换行符 - 希望无论如何都不应该出现在文件名中)。
注意
prename
for portability to的使用ksh
,其中rename
实际上是一个内置命令,而不是独立的perl
可执行文件。只需使用
rename 's/\b(\w)/\u$1/' *
没有 g 标志。g 标志表示“全局 = 多次执行”。您第一次在文件名中输入,而您不想在第二次输入大写,对吧?