使用 prename 或 sed 搜索和替换

Search and replace using prename or sed

我正在尝试按如下方式在文件名中执行搜索和替换:

$ tree
.
├── a_a
│   ├── a_b_c
│   ├── a b c.mkv
│   ├── b_b
│   └── b_c_d
├── b_a
│   ├── a_f_r
│   ├── c_d
│   ├── f_r_e
│   └── r r
├── c_r
│   ├── d_f
│   └── r_a_s.mkv
└── d.mkv

我想用空格替换文件和文件夹名称中的下划线。我想要这样做的方法是首先替换内部目录中存在的文件和文件夹的基本名称中的下划线,然后向上移动,以便我递归的路径在下一次迭代中仍然存在,因为如果我重命名上层目录,在下一次迭代中访问其内部目录和文件的路径将失效。

我知道我可以使用 find 命令对文件进行递归。现在我想用一个工具从里面的文件开始,然后向外移动进行替换操作。我在编写正则表达式方面没有太多经验,但我认为我们可以使用正则表达式中的分组来做到这一点,但我不确定,所以请帮忙。

到现在为止我已经能够弄清楚我们可以使用正则表达式组来访问文件名的某些部分。更具体地说,我们可以使用以下正则表达式获取文件夹和文件的基本名称:

rename -n 's!([^/]*\Z)!uc()!e' ./*

在重命名命令中使用上面的正则表达式,我们可以将基本名称组转换为大写,我想知道如何将该组中的下划线替换为空格。

PS:另外我知道你们中的一些人可能会说这是一个重复的问题,但请再读一遍,我在问这个问题之前已经做了很多研究,但在任何地方都找不到这个具体的问题。

#!/bin/bash

# man find, search for -type
#
# these are other types:
# b - block special, c - character special, d - directory, p - named pipe
# f - regular file, l - symbolic link, s - socket

# Move directories first, then everything else
for TYPE in d f; do

    for NAME in $( find . -type $TYPE -print0 ); do
        if [[ $NAME =~ [a-z] ]]; then
            NEW_NAME=$NAME
            NEW_NAME=${NEW_NAME//[\_]/-} # Change '-' to ' ' if you insist on spaces
            echo "renaming '$NAME' to '$NEW_NAME'"
            mv "$NAME" "$NEW_NAME"
        fi
    done

done
#!/bin/bash
find -depth | while IFS= read -r fn; do
    pnew=$(dirname  "$fn") 
    fnew=$(basename "$fn") 
    if [[ "$fnew" =~ "_" ]]; then
        new="$pnew/${fnew//_/ }"
        echo "$fn -> $new"  
        mv "$fn" "$new"
    fi
done

备注:

  • -depth 参数让 find 遍历目录深度优先。
  • dirname/basename-split 防止目录与其子目录一起重命名。一次只能重命名叶子 file/directory。
  • 所有内容都在需要的地方被引用以允许文件名中有空格(包括传入的文件名)。