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 / 问题 / 459367
Accepted
Florian Brucker
Florian Brucker
Asked: 2018-07-31 05:52:28 +0800 CST2018-07-31 05:52:28 +0800 CST 2018-07-31 05:52:28 +0800 CST

将 shell 变量用于命令选项

  • 772

在 Bash 脚本中,我试图将我正在使用的选项存储rsync在一个单独的变量中。这适用于简单的选项(如--recursive),但我遇到了问题--exclude='.*':

$ find source
source
source/.bar
source/foo

$ rsync -rnv --exclude='.*' source/ dest
sending incremental file list
foo

sent 57 bytes  received 19 bytes  152.00 bytes/sec
total size is 0  speedup is 0.00 (DRY RUN)

$ RSYNC_OPTIONS="-rnv --exclude='.*'"

$ rsync $RSYNC_OPTIONS source/ dest
sending incremental file list
.bar
foo

sent 78 bytes  received 22 bytes  200.00 bytes/sec
total size is 0  speedup is 0.00 (DRY RUN)

As you can see, passing --exclude='.*'to rsync"manually" works fine ( .barisn't copied), it doesn't work when the options are stored in a variable first.

我猜这与引号或通配符(或两者)有关,但我无法弄清楚到底是什么问题。

bash variable
  • 3 3 个回答
  • 26257 Views

3 个回答

  • Voted
  1. Best Answer
    Kusalananda
    2018-07-31T05:55:22+08:002018-07-31T05:55:22+08:00

    通常,将单独的项目列表降级为单个字符串是一个坏主意,无论它是命令行选项列表还是路径名列表。

    改用数组:

    rsync_options=( -rnv --exclude='.*' )
    

    或者

    rsync_options=( -r -n -v --exclude='.*' )
    

    然后...

    rsync "${rsync_options[@]}" source/ target
    

    这样,单个选项的引用就被保持(只要你双引号 的扩展${rsync_options[@]})。它还允许您在调用rsync.

    在任何 POSIX shell 中,可以为此使用位置参数列表:

    set -- -rnv --exclude='.*'
    
    rsync "$@" source/ target
    

    同样,在这里双引号的扩展$@是至关重要的。

    切线相关:

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

    问题是当您将两组选项放入一个字符串时,--exclude选项值的单引号将成为该值的一部分。因此,

    RSYNC_OPTIONS='-rnv --exclude=.*'
    

    本来可以工作¹...但最好(更安全)使用带有单独引用条目的数组或位置参数。如果需要,这样做还允许您使用其中带有空格的内容,并避免让 shell 对选项执行文件名生成(通配符)。


    ¹ 前提$IFS是未修改且--exclude=.当前目录中没有名称以开头的文件,并且未设置nullglobor shell 选项。failglob

    • 66
  2. Florian Brucker
    2018-07-31T23:36:38+08:002018-07-31T23:36:38+08:00

    @Kusalananda已经解释了基本问题以及如何解决它,@glenn jackmann 链接到的Bash FAQ 条目也提供了很多有用的信息。以下是根据这些资源对我的问题中发生的事情的详细解释。

    我们将使用一个小脚本将其每个参数打印在单独的行上来说明事情(argtest.bash):

    #!/bin/bash
    
    for var in "$@"
    do
        echo "$var"
    done
    

    “手动”传递选项:

    $ ./argtest.bash -rnv --exclude='.*'
    -rnv
    --exclude=.*
    

    正如预期的那样,部分-rnv和--exclude='.*'被分成两个参数,因为它们由不带引号的空格分隔(这称为分词)。

    另请注意,周围的引号.*已被删除:单引号告诉 shell 传递其内容而无需特殊解释,但引号本身并未传递给 command。

    如果我们现在将选项作为字符串存储在变量中(而不是使用数组),那么引号不会被删除:

    $ OPTS="--exclude='.*'"
    
    $ ./argtest.bash $OPTS
    --exclude='.*'
    

    这是因为两个原因:定义时使用的双引号$OPTS防止对单引号进行特殊处理,因此后者是值的一部分:

    $ echo $OPTS
    --exclude='.*'
    

    当我们现在将$OPTS其用作命令的参数时,引号在参数扩展之前被处理,因此引号中的$OPTS出现“为时已晚”。

    这意味着(在我原来的问题中)rsync使用排除模式'.*'(带引号!)而不是模式.*- 它排除名称以单引号后跟一个点并以单引号结尾的文件。显然这不是本意。

    一种解决方法是在定义时省略双引号$OPTS:

    $ OPTS2=--exclude='.*'
    
    $ ./argtest.bash $OPTS2
    --exclude=.*
    

    但是,由于在更复杂的情况下存在细微差别,因此始终引用变量赋值是一种好习惯。

    正如@Kusalananda 指出的那样,不引用.*也可以。我添加了引号以防止模式扩展,但在这种特殊情况下这并不是绝对必要的:

    $ ./argtest.bash --exclude=.*
    --exclude=.*
    

    事实证明,Bash确实执行了模式扩展,但模式--exclude=.*与任何文件都不匹配,因此模式被传递给命令。相比:

    $ touch some_file
    
    $ ./argtest.bash some_*
    some_file
    
    $ ./argtest.bash does_not_exit_*
    does_not_exit_*
    

    但是,不引用模式是危险的,因为如果(无论出于何种原因)有一个文件匹配--exclude=.*,那么模式就会被扩展:

    $ touch -- --exclude=.special-filenames-happen
    
    $ ./argtest.bash --exclude=.*
    --exclude=.special-filenames-happen
    

    最后,让我们看看为什么使用数组可以防止我的引用问题(除了使用数组存储命令参数的其他优点之外)。

    定义数组时,分词和引号处理按预期进行:

    $ ARRAY_OPTS=( -rnv --exclude='.*' )
    
    $ echo length of the array: "${#ARRAY_OPTS[@]}"
    length of the array: 2
    
    $ echo first element: "${ARRAY_OPTS[0]}"
    first element: -rnv
    
    $ echo second element: "${ARRAY_OPTS[1]}"
    second element: --exclude=.*
    

    将选项传递给命令时,我们使用语法"${ARRAY[@]}",它将数组的每个元素扩展为一个单独的单词:

    $ ./argtest.bash "${ARRAY_OPTS[@]}"
    -rnv
    --exclude=.*
    
    • 6
  3. champion-runner
    2019-11-20T04:09:44+08:002019-11-20T04:09:44+08:00

    当我们编写函数和 shell 脚本时,传入参数进行处理,参数将传递给 int 数字命名的变量,例如 $1, $2, $3

    例如:

    bash my_script.sh Hello 42 World

    在内部my_script.sh,命令将用于$1引用 Hello、$2to42和$3forWorld

    变量引用 ,$0将扩展为当前脚本的名称,例如my_script.sh

    不要使用命令作为变量来播放整个代码。

    请记住:

    1 避免在脚本中使用全大写的变量名。

    2 不要使用反引号,使用 $(...) 代替,它嵌套更好。

    if [ $# -ne 2 ]
    then
        echo "Usage: $(basename $0) DIRECTORY BACKUP_DIRECTORY"
        exit 1
    fi
    
    directory=$1
    backup_directory=$2
    current_date=$(date +%Y-%m-%dT%H-%M-%S)
    backup_file="${backup_directory}/${current_date}.backup"
    
    tar cv "$directory" | openssl des3 -salt | split -b 1024m - "$backup_file"
    
    • 0

相关问题

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

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

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

Sidebar

Stats

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

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

    • 4 个回答
  • Marko Smith

    ssh 无法协商:“找不到匹配的密码”,正在拒绝 cbc

    • 4 个回答
  • Marko Smith

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

    • 5 个回答
  • Marko Smith

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

    • 3 个回答
  • Marko Smith

    如何卸载内核模块“nvidia-drm”?

    • 13 个回答
  • 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
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Wong Jia Hau ssh-add 返回:“连接代理时出错:没有这样的文件或目录” 2018-08-24 23:28:13 +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
  • Martin Hope
    Bagas Sanjaya 为什么 Linux 使用 LF 作为换行符? 2017-12-20 05:48:21 +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