我必须递归地重命名一个完整的文件夹树,这样任何地方都不会出现大写字母(它是C ++源代码,但这无关紧要)。
忽略CVS和Subversion版本控制文件/文件夹的加分点。首选方法是shell脚本,因为shell应该可以在任何Linux机器上使用。
关于文件重命名的细节有一些有效的论据。
使用"rename"
命令的简洁版本:
find my_root_dir -depth -exec rename 's/(.*)\/([^\/]*)/$1\/\L$2/' {} \;
这避免了在文件之前重命名目录并尝试将文件移动到不存在的目录(例如"A/A"
到"a/a"
)的问题。
或者,更简洁的版本,而不使用"rename"
。
for SRC in `find my_root_dir -depth`
do
DST=`dirname "${SRC}"`/`basename "${SRC}" | tr '[A-Z]' '[a-z]'`
if [ "${SRC}" != "${DST}" ]
then
[ ! -e "${DST}" ] && mv -T "${SRC}" "${DST}" || echo "${SRC} was not renamed"
fi
done
后者允许使用move命令更灵活(例如,"svn mv"
)。
这是我的次优解决方案,使用Bash shell脚本:
#!/bin/bash
# First, rename all folders
for f in `find . -depth ! -name CVS -type d`; do
g=`dirname "$f"`/`basename "$f" | tr '[A-Z]' '[a-z]'`
if [ "xxx$f" != "xxx$g" ]; then
echo "Renaming folder $f"
mv -f "$f" "$g"
fi
done
# Now, rename all files
for f in `find . ! -type d`; do
g=`dirname "$f"`/`basename "$f" | tr '[A-Z]' '[a-z]'`
if [ "xxx$f" != "xxx$g" ]; then
echo "Renaming file $f"
mv -f "$f" "$g"
fi
done
文件夹全部正确重命名,mv
在权限不匹配时不询问问题,并且不重命名CVS文件夹(遗憾的是,该文件夹中的CVS控制文件仍然被重命名)。
由于“find -depth”和“find | sort -r”都以可用的顺序返回文件夹列表以进行重命名,因此我更喜欢使用“-depth”来搜索文件夹。
使用Larry Wall的文件名修复器:
$op = shift or die $help;
chomp(@ARGV = <STDIN>) unless @ARGV;
for (@ARGV) {
$was = $_;
eval $op;
die $@ if $@;
rename($was,$_) unless $was eq $_;
}
这很简单
find | fix 'tr/A-Z/a-z/'
(其中fix
当然是上面的脚本)
这是一个小shell脚本,可以满足您的要求:
root_directory="${1?-please specify parent directory}"
do_it () {
awk '{ lc= tolower($0); if (lc != $0) print "mv \"" $0 "\" \"" lc "\"" }' | sh
}
# first the folders
find "$root_directory" -depth -type d | do_it
find "$root_directory" ! -type d | do_it
请注意第一个查找中的-depth操作。
之前发布的内容可以完美地开箱即用,或者针对简单案例进行一些调整,但在运行批量重命名之前,您可能需要考虑一些情况:
ABCdef
,abcDEF
和aBcDeF
,会发生什么?重命名脚本应该中止还是只是警告并继续?for f in `find -depth`; do mv ${f} ${f,,} ; done
find -depth
打印每个文件和目录,目录的内容打印在目录本身之前。 ${f,,}
小写文件名。
使用MacOS,
安装rename
包,
brew install rename
采用,
find . -iname "*.py" -type f | xargs -I% rename -c -f "%"
此命令查找具有*.py
扩展名的所有文件,并将文件名转换为小写。
`f` - forces a rename
例如,
$ find . -iname "*.py" -type f
./sample/Sample_File.py
./sample_file.py
$ find . -iname "*.py" -type f | xargs -I% rename -c -f "%"
$ find . -iname "*.py" -type f
./sample/sample_file.py
./sample_file.py
不便携,仅限Zsh,但非常简洁。
首先,确保加载zmv
。
autoload -U zmv
另外,请确保extendedglob
已开启:
setopt extendedglob
然后使用:
zmv '(**/)(*)~CVS~**/CVS' '${1}${(L)2}'
以递归方式小写名称不是CVS的文件和目录。
冗长,但“没有惊喜,没有安装工作”
此脚本处理带有空格,引号,其他异常字符和Unicode的文件名,适用于不区分大小写的文件系统和大多数安装了bash和awk的Unix-y环境(即几乎所有)。它还会报告冲突(如果有的话)(将文件名保留为大写),当然也会重命名文件和目录,并以递归方式工作。最后它具有很强的适应性:你可以调整find命令来定位你想要的文件/目录,你可以调整awk来做其他的名字操作。请注意,通过“处理Unicode”我的意思是它确实会转换它们的情况(不要忽略它们像使用tr
的答案)。
# adapt the following command _IF_ you want to deal with specific files/dirs
find . -depth -mindepth 1 -exec bash -c '
for file do
# adapt the awk command if you wish to rename to something other than lowercase
newname=$(dirname "$file")/$(basename "$file" | awk "{print tolower(\$0)}")
if [ "$file" != "$newname" ] ; then
# the extra step with the temp filename is for case-insensitive filesystems
if [ ! -e "$newname" ] && [ ! -e "$newname.lcrnm.tmp" ] ; then
mv -T "$file" "$newname.lcrnm.tmp" && mv -T "$newname.lcrnm.tmp" "$newname"
else
echo "ERROR: Name already exists: $newname"
fi
fi
done
' sh {} +
参考
我的脚本基于以下优秀答案:
https://unix.stackexchange.com/questions/9496/looping-through-files-with-spaces-in-the-names
我需要在Windows 7上的Cygwin设置上执行此操作,并发现我在上面尝试的语法错误(尽管我可能错过了一个工作选项)。然而,这个直接来自Ubuntu forums的解决方案可以解决问题:-)
ls | while read upName; do loName=`echo "${upName}" | tr '[:upper:]' '[:lower:]'`; mv "$upName" "$loName"; done
(注意:我之前用下划线替换了空格:
for f in *\ *; do mv "$f" "${f// /_}"; done
)
在OS X中,mv -f
显示“相同文件”错误,因此我重命名两次:
for i in `find . -name "*" -type f |grep -e "[A-Z]"`; do j=`echo $i | tr '[A-Z]' '[a-z]' | sed s/\-1$//`; mv $i $i-1; mv $i-1 $j; done
在这种情况下我会找到Python,以避免乐观地假设没有空格或斜线的路径。我还发现python2
往往安装在比rename
更多的地方。
#!/usr/bin/env python2
import sys, os
def rename_dir(directory):
print('DEBUG: rename('+directory+')')
# Rename current directory if needed
os.rename(directory, directory.lower())
directory = directory.lower()
# Rename children
for fn in os.listdir(directory):
path = os.path.join(directory, fn)
os.rename(path, path.lower())
path = path.lower()
# Rename children within, if this child is a directory
if os.path.isdir(path):
rename_dir(path)
# Run program, using the first argument passed to this Python script as the name of the folder
rename_dir(sys.argv[1])
我在Mac OS X上找到的最简单的方法是使用http://plasmasturm.org/code/rename/的重命名包:
brew install rename
rename --force --lower-case --nows *
--force即使具有目标名称的文件已存在,也会重命名。
--lower-case将文件名转换为全小写。
--nows用单个下划线字符替换文件名中的所有空格序列。
( find YOURDIR -type d | sort -r;
find yourdir -type f ) |
grep -v /CVS | grep -v /SVN |
while read f; do mv -v $f `echo $f | tr '[A-Z]' '[a-z]'`; done
首先将目录重命名为sort -r(其中-depth不可用),然后是文件。然后grep -v / CVS而不是找到...-修剪因为它更简单。对于大型目录,对于f in ...可以溢出一些shell缓冲区。使用find ... |阅读时要避免这种情况。
是的,这将是clobber文件,只有在案件中有所不同...
这也适用于macOS:
ruby -e "Dir['*'].each { |p| File.rename(p, p.downcase) }"
find . -depth -name '*[A-Z]*'|sed -n 's/\(.*\/\)\(.*\)/mv -n -v -T \1\2 \1\L\2/p'|sh
我没有尝试过这里提到的更复杂的脚本,但是我的Synology NAS上没有一个单一的命令行版本适用于我。 rename
不可用,并且find
的许多变体失败,因为它似乎坚持已经重命名的路径的旧名称(例如,如果它发现./FOO
后跟./FOO/BAR
,将./FOO
重命名为./foo
仍将继续列出./FOO/BAR
,即使该路径不再有效)。以上命令对我没有任何问题。
以下是对命令各部分的解释:
find . -depth -name '*[A-Z]*'
这将从当前目录中找到任何文件(将.
更改为您要处理的任何目录),使用深度优先搜索(例如,它将在./foo/bar
之前列出./foo
),但仅适用于包含大写字符的文件。 -name
过滤器仅适用于基本文件名,而不适用于完整路径。所以这将列出./FOO/BAR
但不是./FOO/bar
。这没关系,因为我们不想重命名./FOO/bar
。我们想重命名./FOO
,但是后面会列出这个(这就是为什么-depth
很重要)。
此命令本身对于首先查找要重命名的文件特别有用。在完成重命名命令后使用此选项可搜索由于文件名冲突或错误而仍未替换的文件。
sed -n 's/\(.*\/\)\(.*\)/mv -n -v -T \1\2 \1\L\2/p'
这部分读取find
输出的文件,并使用正则表达式在mv
命令中格式化它们。 -n
选项阻止sed
打印输入,搜索和替换正则表达式中的p
命令输出替换的文本。
正则表达式本身包含两个捕获:直到最后一个/(文件的目录)的部分,以及文件名本身。目录保持不变,但文件名转换为小写。所以,如果find
输出./FOO/BAR
,它将成为mv -n -v -T ./FOO/BAR ./FOO/bar
。 -n
的mv
选项可确保不会覆盖现有的小写文件。 -v
选项使mv
输出它所做的每一次改变(或者不做 - 如果./FOO/bar
已经存在,它输出类似./FOO/BAR -> ./FOO/BAR
的东西,注意到没有做出任何改变)。 -T
在这里非常重要 - 它将目标文件视为目录。如果该目录恰好存在,这将确保./FOO/BAR
不会移动到./FOO/bar
。
与find
一起使用它可以生成一个将要执行的命令列表(方便验证将在不实际执行的情况下执行的操作)
sh
这很不言自明。它将生成的所有mv
命令路由到shell解释器。你可以用bash
或你喜欢的任何外壳替换它。
这不是OP所要求的,但我希望在此页面上找到的内容:
用于重命名文件的“slugify”版本,因此它们与URL类似(即仅包括字母数字,点和短划线):
rename "s/[^a-zA-Z0-9\.]+/-/g" filename
如果你使用Arch Linux,你可以从rename
安装AUR
包,它提供renamexm
命令作为/usr/bin/renamexm
可执行文件和a manual page。
它是一个非常强大的工具,可以快速重命名文件和目录。
rename -l Developers.mp3 # or --lowcase
rename -u developers.mp3 # or --upcase, long option
-R --recursive # directory and its children
-t --test # Dry run, output but don't rename
-o --owner # Change file owner as well to user specified
-v --verbose # Output what file is renamed and its new name
-s/str/str2 # Substitute string on pattern
--yes # Confirm all actions
如果需要,您可以从here获取示例Developers.mp3文件;)
for f in `find`; do mv -v "$f" "`echo $f | tr '[A-Z]' '[a-z]'`"; done
如果您不需要关心效率,只需尝试以下操作即可。
zip -r foo.zip foo/*
unzip -LL foo.zip
上面的大多数答案都很危险,因为它们不涉及包含奇数字符的名称。对于这类事情,你最安全的选择是使用find
的-print0
选项,它将使用ASCII NUL而不是\ n来终止文件名。
这是一个脚本,它只改变文件而不是目录名,以免混淆find
:
find . -type f -print0 | xargs -0n 1 bash -c \
's=$(dirname "$0")/$(basename "$0");
d=$(dirname "$0")/$(basename "$0"|tr "[A-Z]" "[a-z]"); mv -f "$s" "$d"'
我测试了它,它适用于包含空格,各种引号等的文件名。这很重要,因为如果你以root身份运行树上其他脚本之一,其中包含由
touch \;\ echo\ hacker::0:0:hacker:\$\'\057\'root:\$\'\057\'bin\$\'\057\'bash
......好吧猜猜......
男人,你们/女孩喜欢过于复杂的事情...使用:
rename 'y/A-Z/a-z/' *
如果您已经拥有或设置了重命名命令(例如通过Mac中的brew install),则此方法有效:
rename --lower-case --force somedir/*
这适用于没有CentOS Perl脚本的Red Hat Linux / rename
或其他发行版:
for i in $( ls | grep [A-Z] ); do mv -i "$i" "`echo $i | tr 'A-Z' 'a-z'`"; done
资料来源:Rename all file names from uppercase to lowercase characters
(在某些发行版中,默认的rename
命令来自util-linux,这是一个不同的,不兼容的工具。)
原始问题要求忽略SVN和CVS目录,这可以通过在find命令中添加-prune来完成。例如,忽略CVS:
find . -name CVS -prune -o -exec mv '{}' `echo {} | tr '[A-Z]' '[a-z]'` \; -print
[编辑]我尝试了这一点,并在查找中嵌入小写翻译不起作用,我实际上并不理解。所以,修改为:
$> cat > tolower
#!/bin/bash
mv $1 `echo $1 | tr '[:upper:]' '[:lower:]'`
^D
$> chmod u+x tolower
$> find . -name CVS -prune -o -exec tolower '{}' \;
伊恩