如何使用bash字符串格式来反转日期格式?

时间:2018-05-10 08:57:20

标签: bash date-manipulation

我有很多名为MM-DD-YYYY.pdf的文件。我想将它们重命名为YYYY-MM-DD.pdf我确信有一些bash魔法可以做到这一点。它是什么?

4 个答案:

答案 0 :(得分:4)

对于当前目录中的文件:

for name in ./??-??-????.pdf; do
    if [[ "$name" =~ (.*)/([0-9]{2})-([0-9]{2})-([0-9]{4})\.pdf ]]; then
        echo mv "$name" "${BASH_REMATCH[1]}/${BASH_REMATCH[4]}-${BASH_REMATCH[3]}-${BASH_REMATCH[2]}.pdf"
    fi
done

递归地,在当前目录中或下面:

find . -type f -name '??-??-????.pdf' -exec bash -c '
    for name do
        if [[ "$name" =~ (.*)/([0-9]{2})-([0-9]{2})-([0-9]{4})\.pdf ]]; then
            echo mv "$name" "${BASH_REMATCH[1]}/${BASH_REMATCH[4]}-${BASH_REMATCH[3]}-${BASH_REMATCH[2]}.pdf"
        fi
    done' bash {} +

globstar中启用bash shell选项可让我们执行以下操作(也将像上述解决方案一样处理当前目录中或下面的所有文件):

shopt -s globstar

for name in **/??-??-????.pdf; do
    if [[ "$name" =~ (.*)/([0-9]{2})-([0-9]{2})-([0-9]{4})\.pdf ]]; then
        echo mv "$name" "${BASH_REMATCH[1]}/${BASH_REMATCH[4]}-${BASH_REMATCH[3]}-${BASH_REMATCH[2]}.pdf"
    fi
done

所有这三个解决方案都使用正则表达式来挑选文件名的相关部分,然后将这些部分重新排列为新名称。它们之间的唯一区别是如何生成路径名列表。

为了安全起见,代码前缀为mvecho。要实际重命名文件,请移除echo(但至少运行 echo以查看它是否符合要求)。

答案 1 :(得分:1)

命令行的直接方法示例:

$ ls
10-01-2018.pdf  11-01-2018.pdf  12-01-2018.pdf
$ ls [0-9]*-[0-9]*-[0-9]*.pdf|sed -r 'p;s/([0-9]{2})-([0-9]{2})-([0-9]{4})/\3-\1-\2/'|xargs -n2 mv
$ ls
2018-10-01.pdf  2018-11-01.pdf  2018-12-01.pdf

ls输出通过管道传输到sed,然后我们使用 p 标志打印参数而不做任何修改,换句话说,就是文件的原始名称,和 s 来执行和输出转换。

ls + sed结果是一个组合输出,由一系列 old_file_name new_file_name 组成。

最后,我们通过xargs管道生成的Feed,以便有效地重命名文件。

来自xargs man

  

-n number使用尽可能多的标准输入参数执行命令,最多为数字参数。

答案 2 :(得分:0)

您可以使用以下命令:klashxx

for f in *.pdf; do echo "$f"; mv "$f" "$(echo "$f" | sed 's@\(..\)-\(..\)-\(....\)@\3-\2-\1@')"; done

<强>之前:

ls *.pdf
12-01-1998.pdf  12-03-2018.pdf

<强>后:

ls *.pdf
1998-01-12.pdf  2018-03-12.pdf

此外,如果您的文件夹中有其他pdf个不符合此格式的文件,您可以做的只是选择符合格式的文件:MM-DD-YYYY.pdf这样做使用以下内容命令:

for f in `find . -maxdepth 1 -type f -regextype sed -regex './[0-9]\{2\}-[0-9]\{2\}-[0-9]\{4\}.pdf' | xargs -n1 basename`; do echo "$f"; mv "$f" "$(echo "$f" | sed 's@\(..\)-\(..\)-\(....\)@\3-\2-\1@')"; done

解释:

  • find . -maxdepth 1 -type f -regextype sed -regex './[0-9]\{2\}-[0-9]\{2\}-[0-9]\{4\}.pdf此find命令将仅查找当前工作目录中尊重语法并提取其基本名称的文件(删除开头的./,文件夹和其他类型的文件不考虑具有相同名称的其他*.pdf文件也会被忽略。

  • for each file你做了一个移动,结果文件名使用sed和back引用来计算MM,DD和YYYY的3组

答案 3 :(得分:0)

对于这些简单的文件名,使用更详细的模式,您可以稍微简化循环体:

twodigit=[[:digit:]][[:digit:]]
fourdigit="$twodigit$twodigit"
for f in $twodigit-$twodigit-$fourdigit.pdf; do
  IFS=- read month day year <<< "${f%.pdf}"
  mv "$f" "$year-$month-$day.pdf"
done

这基本上是@Kusalananda's answer,但没有正则表达式匹配的详细程度。