如何在linux上更改包含特定字符串的所有目录/文件的名称

时间:2011-06-02 18:01:16

标签: linux shell unix pipe

我有一个目录,我想将其下的所有目录名称和文件更改为其他名称。例如我的目录结构是

./mydir_ABC/
./mydir_ABC/myfile_ABC.txt
./mydir_ABC/otherdir_ABC/

我希望以此为例

./mydir_DEF/
./mydir_DEF/myfile_DEF.txt
./mydir_DEF/otherdir_DEF/

我正在使用

find . -type 'f' -name '*ABC*'
find . -type 'd' -name '*ABC*'

获取名称中包含我的字符串的所有相关目录和文件的列表。如何将其传递给另一个实际更改目录名称和文件名的命令? (如果文件或目录已经存在,我希望它能够覆盖。)

3 个答案:

答案 0 :(得分:2)

mkdir z; cd z; touch fooabcbar; touch fooABCbar
find . -type 'f' \( -name '*ABC*' -o -name '*abc*' \) | while read f; do g=`echo "$f" | sed -e 's/abc/def/g' -e 's/ABC/DEF/g'`; echo mv -- "$f" "$g"; done

只要路径名不包含换行符(非常罕见)。

答案 1 :(得分:2)

查找rename(1)命令。它有一些不同的变体,但它们都支持基于某种正则表达式的某种重命名。我使用的版本(基于第一版Perl'Camel Book'的代码)将用作:

rename 's%ABC%DEF%g' ...

或者,对于您的示例:

find . -print0 | xargs -0 rename s/ABC/DEF/g

或者,为了避免在遍历目录结构时更改路径名称的问题,请首先执行目录名称,然后执行文件(如问题中所示):

find . -type d -print0 | xargs -0 rename s/ABC/DEF/g
find . -type f -print0 | xargs -0 rename s/ABC/DEF/g

在您继续重命名目录时仍然存在问题(因为一旦您将./ABC重命名为./DEF,您就无法再将./ABC/subABC/重命名为./DEF/subDEF,因为目录现在是./DEF/subABC)。

其他版本的rename具有不同的语法 - 请查看您的手册页。


这是基于Perl的版本rename,源自第一版Perl'Camel Book';我在版本控制下的第一个版本的日期是1992-01-05。今天的更改更新了shebang行(使用env,已移除-w,因为它不可靠,并添加use warnings;以补偿不-w)。先前的变化是2008年,1998年,1996年和1992年;在过去的19年里,它没有太大变化。

#!/usr/bin/env perl
#
# @(#)$Id: rename.pl,v 1.8 2011/06/03 22:30:22 jleffler Exp $
#
# Rename files using a Perl substitute or transliterate command

use strict;
use warnings;
use Getopt::Std;

my(%opts);
my($usage) = "Usage: $0 [-fnxV] perlexpr [filenames]\n";
my($force) = 0;
my($noexc) = 0;
my($trace) = 0;

die $usage unless getopts('fnxV', \%opts);

if ($opts{V})
{
    printf "%s\n", q'RENAME Version $Revision: 1.8 $ ($Date: 2011/06/03 22:30:22 $)';
    exit 0;
}
$force = 1 if ($opts{f});
$noexc = 1 if ($opts{n});
$trace = 1 if ($opts{x});

my($op) = shift;
die $usage unless defined $op;

if (!@ARGV) {
    @ARGV = <STDIN>;
    chop(@ARGV);
}

for (@ARGV)
{
    if (-e $_ || -l $_)
    {
        my($was) = $_;
        eval $op;
        die $@ if $@;
        next if ($was eq $_);
        if ($force == 0 && -f $_)
        {
            print STDERR "rename failed: $was - $_ exists\n";
        }
        else
        {
            print "+ $was --> $_\n" if $trace;
            print STDERR "rename failed: $was - $!\n"
                unless ($noexc || rename($was, $_));
        }
    }
    else
    {
        print STDERR "$_ - $!\n";
    }
}

答案 2 :(得分:0)

如果它始终是最后3个字符,那么它应该运行得很好我会想:

mv *abc.* *def.*