如何在Linux上将所有文件夹和文件重命名为小写?

问题描述 投票:153回答:27

我必须递归地重命名一个完整的文件夹树,这样任何地方都不会出现大写字母(它是C ++源代码,但这无关紧要)。

忽略CVS和Subversion版本控制文件/文件夹的加分点。首选方法是shell脚本,因为shell应该可以在任何Linux机器上使用。

关于文件重命名的细节有一些有效的论据。

  1. 我认为应该覆盖具有相同小写名称的文件;这是用户的问题。当在一个忽略大小写的文件系统上签出时,它也会用后者覆盖第一个。
  2. 我会考虑A-Z字符并将它们转换为a-z,其他一切只是要求问题(至少使用源代码)。
  3. 在Linux系统上运行构建需要脚本,因此我认为应该省略对CVS或Subversion版本控制文件的更改。毕竟,这只是一个划痕结账。也许“出口”更合适。
linux rename lowercase
27个回答
161
投票

使用"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

P.S.

后者允许使用move命令更灵活(例如,"svn mv")。


4
投票

这是我的次优解决方案,使用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”来搜索文件夹。


4
投票

使用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当然是上面的脚本)


3
投票

这是一个小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操作。


2
投票

之前发布的内容可以完美地开箱即用,或者针对简单案例进行一些调整,但在运行批量重命名之前,您可能需要考虑一些情况:

  1. 如果在路径层次结构中同一级别有两个或多个名称,只有大小写不同,例如ABCdefabcDEFaBcDeF,会发生什么?重命名脚本应该中止还是只是警告并继续?
  2. 如何为非US-ASCII名称定义小写?如果可能存在此类名称,是否应首先执行检查和排除传递?
  3. 如果在CVS或SVN工作副本上运行重命名操作,则更改文件或目录名称的大小写可能会损坏工作副本。该脚本是否还应查找并调整内部管理文件,如.svn / entries或CVS / Entries?

2
投票
for f in `find -depth`; do mv ${f} ${f,,} ; done

find -depth打印每个文件和目录,目录的内容打印在目录本身之前。 ${f,,}小写文件名。


2
投票

使用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

1
投票

不便携,仅限Zsh,但非常简洁。

首先,确保加载zmv

autoload -U zmv

另外,请确保extendedglob已开启:

setopt extendedglob

然后使用:

zmv '(**/)(*)~CVS~**/CVS' '${1}${(L)2}'

以递归方式小写名称不是CVS的文件和目录。


1
投票

冗长,但“没有惊喜,没有安装工作”

此脚本处理带有空格,引号,其他异常字符和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

How to convert a string to lower case in Bash?


1
投票

我需要在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

)


1
投票

在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

241
投票

更小我仍然喜欢:

rename 'y/A-Z/a-z/' *

在不区分大小写的文件系统(如OS X的HFS+)上,您需要添加-f标志:

rename -f 'y/A-Z/a-z/' *

1
投票

在这种情况下我会找到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])

1
投票

我在Mac OS X上找到的最简单的方法是使用http://plasmasturm.org/code/rename/的重命名包:

brew install rename
rename --force --lower-case --nows *

--force即使具有目标名称的文件已存在,也会重命名。

--lower-case将文件名转换为全小写。

--nows用单个下划线字符替换文件名中的所有空格序列。


0
投票
( 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文件,只有在案件中有所不同...


0
投票

这也适用于macOS:

ruby -e "Dir['*'].each { |p| File.rename(p, p.downcase) }"

0
投票
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-nmv选项可确保不会覆盖现有的小写文件。 -v选项使mv输出它所做的每一次改变(或者不做 - 如果./FOO/bar已经存在,它输出类似./FOO/BAR -> ./FOO/BAR的东西,注意到没有做出任何改变)。 -T在这里非常重要 - 它将目标文件视为目录。如果该目录恰好存在,这将确保./FOO/BAR不会移动到./FOO/bar

find一起使用它可以生成一个将要执行的命令列表(方便验证将在不实际执行的情况下执行的操作)


sh

这很不言自明。它将生成的所有mv命令路由到shell解释器。你可以用bash或你喜欢的任何外壳替换它。


0
投票

Slugify Rename (regex)

这不是OP所要求的,但我希望在此页面上找到的内容:

用于重命名文件的“slugify”版本,因此它们与URL类似(即仅包括字母数字,点和短划线):

rename "s/[^a-zA-Z0-9\.]+/-/g" filename

0
投票

如果你使用Arch Linux,你可以从rename安装AUR包,它提供renamexm命令作为/usr/bin/renamexm可执行文件和a manual page

它是一个非常强大的工具,可以快速重命名文件和目录。

Convert to lowercase

rename -l Developers.mp3 # or --lowcase

Convert to UPPER case

rename -u developers.mp3 # or --upcase, long option

Other options

-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文件;)


0
投票

使用typeset

typeset -l new        # Always lowercase
find $topPoint |      # Not using xargs to make this more readable
  while read old
  do mv "$old" "$new" # Quotes for those annoying embedded spaces
  done

在Windows上,像Git Bash这样的仿真可能会失败,因为Windows在引擎盖下不区分大小写。对于那些,首先添加一个mv文件的步骤到另一个名称,如“$ old.tmp”,然后再添加到$new


83
投票
for f in `find`; do mv -v "$f" "`echo $f | tr '[A-Z]' '[a-z]'`"; done

74
投票

如果您不需要关心效率,只需尝试以下操作即可。

zip -r foo.zip foo/*
unzip -LL foo.zip

14
投票

上面的大多数答案都很危险,因为它们不涉及包含奇数字符的名称。对于这类事情,你最安全的选择是使用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

......好吧猜猜......


13
投票

男人,你们/女孩喜欢过于复杂的事情...使用:

rename 'y/A-Z/a-z/' *

12
投票

如果您已经拥有或设置了重命名命令(例如通过Mac中的brew install),则此方法有效:

rename --lower-case --force somedir/*

7
投票

这适用于没有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,这是一个不同的,不兼容的工具。)


4
投票

原始问题要求忽略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 '{}'  \;

伊恩

© www.soinside.com 2019 - 2024. All rights reserved.