在Bash或Perl中重命名和移动文件

5

您好,我完全不懂Bash和StackOverflow。

我需要将一组文件(全部包含在同一个文件夹中)移动到目标文件夹,其中可能已经存在具有相同名称的文件。

如果特定文件存在,则需要在移动文件之前重命名该文件,例如通过将递增整数附加到文件名。

扩展名应保留(换句话说,该附加的递增整数应在扩展名之前)。文件名可能在中间包含点。

最初,我想比较这两个文件夹,以获得现有文件的列表(我使用了“comm”),但后来我有些卡住了。我认为我只是试图以最复杂的方式做事情。

有没有提示以“bash方式”完成此操作?如果在除bash脚本之外的脚本中完成也可以。


@Katie - 我擅自编辑了你的问题,并在其中加入了你在评论中澄清的两个事项。 - DVK
5个回答

7
如果您不介意重命名已经存在的文件,GNU mv 命令有一个 --backup 选项:
mv --backup=numbered * /some/other/dir

实际上,我需要保留目标目录中文件的名称。此外,我应该保留扩展名。但是感谢这个提示,它在其他情况下可能会有用... - Katie

4

这是一个Bash脚本:

source="/some/dir"
dest="/another/dir"
find "$source" -maxdepth 1 -type f -printf "%f\n" | while read -r file
do
    suffix=
    if [[ -a "$dest/$file" ]]
    then
        suffix=".new"
    fi
    # to make active, comment out the next line and uncomment the line below it
    echo 'mv' "\"$source/$file\"" "\"$dest/$file$suffix\""
    # mv "source/$file" "$dest/$file$suffix"
 done

后缀名会被盲目地添加。如果两个目录中都有名为“foo.new”的文件,则结果将是一个名为“foo.new”的文件和第二个名为“foo.new.new”的文件,这可能看起来很傻,但正确的做法是不覆盖文件。但是,如果目标已经包含“foo.new.new”(并且源和目标中都有“foo.new”),则“foo.new.new”将被覆盖。
您可以将上面的if更改为循环以处理该情况。此版本还保留了扩展名。
source="/some/dir"
dest="/another/dir"
find "$source" -maxdepth 1 -type f -printf "%f\n" | while read -r file
do
    suffix=
    count=
    ext=
    base="${file%.*}"
    if [[ $file =~ \. ]]
    then
        ext=".${file##*.}"
    fi
    while [[ -a "$dest/$base$suffix$count$ext" ]]
    do
        (( count+=1 ))
        suffix="."
    done
    # to make active, comment out the next line and uncomment the line below it
    echo 'mv' "\"$source/$file\"" "\"$dest/$file$suffix$count$ext\""
    # mv "$source/$file" "$dest/$file$suffix$count$ext"
done

@Dennis - 很好的做法注释掉 'mv' :) - DVK

3
根据 OP 的说法,这并不仅限于 bash,还包括 Perl。下面是新的解决方案(注意扩展名)。
~/junk/a1$ ls
f1.txt   f2.txt   f3.txt   z1       z2


~/junk/a1$ ls ../a2
f1.txt     f2.1.txt   f2.2.txt   f2.3.txt   f2.txt     z1

# I split the one-liner into multiple lines for readability
$ perl5.8 -e 
     '{use strict; use warnings; use File::Copy; use File::Basename; 
       my @files = glob("*"); # assume current directory
       foreach my $file (@files) {
           my $file_base2 = basename($file); 
           my ($file_base, $ext) = ($file_base2 =~ /(.+?)([.][^.]+$)?$/);
           my $new_file_base = "../a2/$file_base";
           my $new_file = $new_file_base . $ext; 
           my $counter = 1;
           while (-e $new_file) { 
               $new_file = "$new_file_base." . $counter++ . $ext;
           }
           copy($file, $new_file)
               || die "could not copy $file to $new_file: $!\n";
        } }'

~/junk/a1> ls ../a2
f1.1.txt f1.txt  f2.1.txt  f2.2.txt  f2.3.txt  f2.4.txt  f2.txt  f3.txt
z1         z1.1       z2

旧方案:(不考虑扩展名)

~/junk/a1$ ls
f1   f2   f3

~/junk/a1$ ls ../a2
f1     f2     f2.1   f2.2   f2.3

# I split the one-liner into multiple lines for readability
$ perl5.8 -e 
     '{use strict; use warnings; use File::Copy; use File::Basename; 
       my @files = glob("*"); # assume current directory
       foreach my $file (@files) {
           my $file_base = basename($file); 
           my $new_file_base = "../a2/$file_base"; 
           my $new_file = $new_file_base; 
           my $counter = 1;
           while (-e $new_file) { $new_file = "$new_file_base." . $counter++; }
           copy($file,$new_file)
               || die "could not copy $file to $new_file: $!\n";
        } }'

~/junk/a1> ls ../a2
f1     f1.1   f2     f2.1   f2.2   f2.3   f2.4   f3

这很棒(我不关心perl或bash),但我应该通过保留扩展名来应用新名称(获取类似于foo_1.txt的foo.txt,而不是foo.text.1)。我正在尝试理解如何做到这一点... - Katie
@Katie - 对于“只要完成工作就行,不在乎细节”的良好态度感到赞赏 :) - DVK
@Katie - 在结尾处将“copy()”调用替换为“move()”,以移动而不是复制。 - DVK
不知道该说什么。它似乎完美无缺。非常感谢你。 - Katie

0

没有经过测试就发布这个内容,我感到很不好。不过现在已经很晚了,明天还要工作。我的尝试大概会是这样:

## copy files from src to dst    
## inserting ~XX into any name between base and extension
## where a name collision would occur
src="$1"
dst="$2"

case "$dst" in
    /*) :;;               # absolute dest is fine
    *)  dst=$(pwd)/$dst;; # relative needs to be fixed up
    esac

cd "$src"
find . -type f | while read x; do
    x=${x#./}           # trim off the ./
    t=$x;               # initial target
    d=$(dirname $x);    # relative directory
    b=$(basename $x);   # initial basename
    ext=${b%%.*};       # extension 
    b=${b##*.};         # basename with ext. stripped off
    let zz=0;           # initial numeric
    while [ -e  "$dst/$t" ]; do
        # target exists, so try constructing a new target name
        t="$d/$bb~$zz.$ext"
        let zz+=1;
    done
    echo mv "./$x" "$dst/$t"
done

总体策略是从源路径获取每个名称,将其分解成部分,并在任何冲突的情况下迭代形式为“base〜XX.extension”的名称,直到找到一个不发生冲突的名称。

显然,我在mv命令之前添加了echo,因为我很胆小。自行删除可能会导致文件受损。


0
如果您不需要增量后缀,rsync 可以完成此工作:
rsync --archive --backup --suffix=.sic src/ dst

更新:

使用find/sed/sort来管理版本化的备份文件:

#!/bin/bash                                                                                                        

src="${1}"
dst="${2}"

if test ! -d "${src}" -o ! -d "${dst}" ;then
    echo Usage: $0 SRC_DIR DST_DIR >&2
    exit 1
fi

rsync --archive --backup "${src}/" "${dst}/"
new_name() {
    local dst=$1
    local prefix=$2
    local suffix=$3
    local max=$(find ${dst} -type f -regex  ".*${prefix}.[0-9]*.${suffix}\$" \
        | sed 's/.*\.\([0-9]*\)\..*/\1/'|sort -n|tail -n 1)
    let max++
    echo ${prefix}.${max}.${suffix}
}

# swap BACKUP-extension/real-extension                                                                             
for backup_file in $(find $dst -name "*~"); do
    file=${backup_file%~}
    prefix=${file%.*}
    suffix=${file##*.}
    suffix=${suffix%\~}
    mv ${backup_file} $(new_name $dst $prefix $suffix)
done

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接