AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 497526
Accepted
Sopalajo de Arrierez
Sopalajo de Arrierez
Asked: 2019-01-30 11:11:24 +0800 CST2019-01-30 11:11:24 +0800 CST 2019-01-30 11:11:24 +0800 CST

Linux shell 脚本:只有存在的程序才运行,不存在则忽略

  • 772

我正在编写一个Linux shell 脚本figlet,该脚本仅在安装了正确的工具(例如:在系统路径上可访问)时才会在其执行期间打印状态横幅。

例子:

#!/usr/bin/env bash
echo "foo"
figlet "Starting"
echo "moo"
figlet "Working"
echo "foo moo"
figlet "Finished"

我希望我的脚本即使在未安装figlet时也能正常工作。

什么是实用的方法?

shell scripting
  • 8 8 个回答
  • 11572 Views

8 个回答

  • Voted
  1. Best Answer
    Jeff Schaller
    2019-01-30T12:20:15+08:002019-01-30T12:20:15+08:00

    我的解释是使用与工具同名的包装函数;在该函数中,如果存在真实工具,则执行它:

    figlet() {
      if command -p figlet  >/dev/null 2>&1
        then 
          command figlet "$@"
        else
          :
        fi
    }
    

    然后你可以figlet arg1 arg2...在你的脚本中保持不变。

    @Olorin 提出了一个更简单的方法:仅在需要时定义包装函数(如果该工具不存在):

    if ! command -v figlet > /dev/null; then figlet() { :; }; fi
    

    如果您希望figlet即使未安装 figlet 也能打印参数,请调整 Olorin 的建议,如下所示:

    if ! command -v figlet > /dev/null; then figlet() { printf '%s\n' "$*"; }; fi
    
    • 38
  2. roaima
    2019-01-30T11:16:54+08:002019-01-30T11:16:54+08:00

    你可以测试看看是否figlet存在

    if type figlet >/dev/null 2>&1
    then
        echo Figlet is installed
    fi
    
    • 16
  3. kasperd
    2019-01-30T13:28:18+08:002019-01-30T13:28:18+08:00

    一种常见的方法是使用test -xaka [ -x。下面是一个取自/etc/init.d/ntpLinux 系统的示例:

    if [ -x /usr/bin/lockfile-create ]; then
        lockfile-create $LOCKFILE
        lockfile-touch $LOCKFILE &
        LOCKTOUCHPID="$!"
    fi
    

    此变体依赖于知道可执行文件的完整路径。在/bin/lesspipe我找到了一个通过组合-x和which命令解决该问题的示例:

    if [ -x "`which bunzip`" ]; then bunzip -c "$1"
    else echo "No bunzip available"; fi ;;
    

    这样就可以在事先不知道可执行文件在哪里PATH的情况下工作。bunzip

    • 7
  4. Chris
    2019-01-31T03:04:05+08:002019-01-31T03:04:05+08:00

    在脚本的开头,检查是否figlet存在,如果不存在,则定义一个什么都不做的 shell 函数:

    type figlet >/dev/null 2>&1 || figlet() { :; }
    

    type检查是否figlet作为 shell 内置、函数、别名或关键字存在,>/dev/null 2>&1丢弃标准输入和标准输出,因此您不会得到任何输出,如果不存在,则figlet() { :; }定义figlet为不执行任何操作的函数。

    这样您就不必编辑使用 的脚本的每一行figlet,或者在每次figlet调用时检查它是否存在。

    如果您愿意,可以添加诊断消息:

    type figlet >/dev/null 2>&1 || { echo 'figlet not installed.' ; figlet() { :; } ; }
    

    作为奖励,由于您没有提及您正在使用哪个外壳,我相信这是符合 POSIX 的,因此它应该适用于大多数外壳。

    • 6
  5. rrauenza
    2019-01-30T13:52:03+08:002019-01-30T13:52:03+08:00

    另一种选择——我在项目自动配置脚本中看到的一种模式:

    if [ -x /usr/bin/figlet ]
    then
        FIGLET=/usr/bin/figlet
    else
        FIGLET=:
    fi
    
    $FIGLET "Hello, world!"
    

    在您的具体情况下,您甚至可以这样做,

    if [ -x /usr/bin/figlet ]
    then
       SAY=/usr/bin/figlet
    elif [ -x /usr/local/bin/figlet ]
    then
       SAY=/usr/local/bin/figlet
    elif [ -x /usr/bin/banner ]
    then
       SAY=/usr/bin/banner
    else
       SAY=/usr/bin/echo
    fi
    
    $SAY "Hello, world!"
    

    如果您不知道具体路径,您可以尝试多个elif(见上文)来尝试已知位置,或者只使用PATH来始终解析命令:

    if command -v figlet >/dev/null
    then
        SAY=figlet
    elif command -v banner >/dev/null
    then
        SAY=banner
    else
        SAY=echo
    fi
    

    一般来说,在编写脚本时,我更喜欢只在我指定的特定位置调用命令。我不喜欢最终用户可能会PATH在他们自己的~/bin.

    例如,如果我正在为其他人编写一个复杂的脚本,该脚本可能会根据我正在调用的特定命令的输出来删除文件,我不想意外地在他们~/bin的命令中找到可能是或可能不是命令的东西我期望。

    • 5
  6. Vercingatorix
    2019-01-30T13:56:31+08:002019-01-30T13:56:31+08:00
    type -p figlet > /dev/null && figlet "foo"
    

    bashtype命令查找命令、函数、别名、关键字或内置命令(请参阅 参考资料help type)并打印出位置或定义。它还返回一个代表搜索结果的返回码;如果找到,则为真 (0)。所以我们在这里所做的是尝试figlet在路径中查找(-p意味着只查找文件,而不是内置函数或函数,并且还抑制错误消息),丢弃输出(就是这样> /dev/null做的),如果它返回 true ( &&) ,它将执行figlet。

    如果figlet在固定位置,这会更简单:

    [ -x /usr/bin/figlet ] && /usr/bin/figlet "foo"
    

    在这里,我们使用test命令 (aka [) 来查看是否/usr/bin/figlet可执行 ( -x),如果是 ( &&) 则执行它。type我认为这个解决方案比使用我认为的 bashism更便携。

    您可以创建一个为您执行此操作的函数:

    function x() {
        if type -p "$1" >/dev/null; then
            cmd="$1"
            shift
            "$cmd" "$@"
        fi
    }
    

    (由于潜在的空间,引号是必要的)

    然后你就这样做:

    x figlet "foo"
    
    • 3
  7. jusdepatate
    2019-01-31T01:49:55+08:002019-01-31T01:49:55+08:00

    o/,我会说类似

    #!/usr/bin/env bash
    # if figlet is installed :
    if [ "$(which figlet 2>/dev/null)" ]; then
           # do what you wanted to do
           echo "foo"
           figlet "Starting"
           echo "moo"
           figlet "Working"
           echo "foo moo"
           figlet "Finished"
    # if not
    else
           # exit program with an error
           echo "please install figlet"
           exit 1
    fi
    
    • 1
  8. nigel222
    2019-02-01T02:50:39+08:002019-02-01T02:50:39+08:00

    您可以执行测试、抑制任何输出并测试成功/失败代码。选择 figlet 的参数以使这个测试变得便宜。-?或 --help 或 --version 是明显的可能性。

    if figlet --help >/dev/null 2>&1 ; then
        # figlet is available
        echo "foo"
        figlet "starting"
        #etc
    else
        rc=$?
        echo "figlet is not installed or not working correctly (return code ${rc})"
    fi
    

    添加以回应下面的评论:如果您真的想测试 figlet 存在,而不是它是否可用,那么您会这样做

    figlet --help >/dev/null 2>&1 
    rc=$?
    if [ $rc -eq 127 ] ; then # 127 is "command not found" on linux bash 4.4.23(1)
        echo "command figlet not found"
    else
        figlet "whatever" # use figlet
    fi
    
    • 0

相关问题

  • FreeBSD 的 sh:列出函数

  • 有没有办法让 ls 只显示某些目录的隐藏文件?

  • grep -v grep 有什么作用

  • 根据其中的值批量重命名(附加)CSV 文件

  • 如何将带有〜的路径保存到变量中?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve