一尘不染

语法错误:“(”意外—在bash脚本中带有!(*。sh)

linux

我想运行一个sh文件:

#!/bin/bash
for f in !(*.sh); do
    ffmpeg -i "$f" -vf yadif=0:-1 -threads 0 -c:v libx264 -pix_fmt yuv420p \
        -r 29.97 -b:v 3000k -s 1280x720 -preset:v slow -profile:v Main \
        -level 3.1 -bf 2 -movflags faststart /mnt/media/out-mp4/"${f%.mxf}.mp4"
    rm $f
done

但是,出现以下错误:

2: task1.sh: Syntax error: "(" unexpected

如果我直接在命令行上尝试,它会完美运行。

路径和权限已被检查

知道会发生什么吗?


阅读 264

收藏
2020-06-02

共1个答案

一尘不染

这不是“ sh文件”,而是 bash 脚本。如果使用来运行它sh yourscript,它将不起作用(因为extixlobs,即您要使用的shell功能,在POSIX sh中不受支持);它需要仅与运行bash yourscript,或用./yourscript与启动时#!/bin/bash(因为它)。因此,将其描述为“
sh文件”会产生误导。而且,即使使用bash,也需要打开扩展的globing功能。


您的直接问题是,!(*.sh)它不是常规的glob语法;它是extglob扩展名,默认情况下不可用。您可能具有一个.bashrc或类似的配置文件,该文件为交互式shell启用了此扩展,但不适用于脚本。跑:

shopt -s extglob

…以启用这些功能。


清理后,您的脚本可能如下所示:

#!/bin/bash

shopt -s extglob

# putting settings in an array allows unescaped newlines in definition
# also sorted to make it easier to find things.
settings=(
  -b:v 3000k
  -bf 2
  -c:v libx264
  -level 3.1
  -movflags faststart
  -pix_fmt yuv420p
  -preset:v slow
  -profile:v Main
  -r 29.97
  -s 1280x720
  -threads 0
  -vf yadif=0:-1
)

for f in !(*.sh); do
    ffmpeg "${settings[@]}" -i "$f" \
      /mnt/media/out-mp4/"${f%.mxf}.mp4" && rm -- "$f"
done

请注意除格式之外的以下更改:

  • shopt -s extglob 在扩展全局范围之前是独立的。
  • rm仅在ffmpeg成功的情况下运行,因为这些命令之间的分隔符为&&,而不是两个;或一个空的换行符。
  • --传递给的参数rm告诉它将所有以后的参数(在本例中为的内容"$f")都视为文件名,即使它以短划线开头也是如此。
  • "$f"参数rm在双引号内。
2020-06-02