如何使用双括号或单括号,括号,花括号


657

我对Bash中括号,括号,花括号的使用以及它们的双精度或单精度之间的区别感到困惑。有明确的解释吗?

Answers:


605

在bash,test并且[是shell内建的。

作为shell关键字的双括号启用了其他功能。例如,您可以使用&&||而不是-aand,-o并且有一个正则表达式匹配运算符=~

同样,在一个简单的测试中,方括号似乎比单个方括号要快得多。

$ time for ((i=0; i<10000000; i++)); do [[ "$i" = 1000 ]]; done

real    0m24.548s
user    0m24.337s
sys 0m0.036s
$ time for ((i=0; i<10000000; i++)); do [ "$i" = 1000 ]; done

real    0m33.478s
user    0m33.478s
sys 0m0.000s

花括号除了用于分隔变量名外,还用于参数扩展,因此您可以执行以下操作:

  • 截断变量的内容

    $ var="abcde"; echo ${var%d*}
    abc
  • 使替换类似于 sed

    $ var="abcde"; echo ${var/de/12}
    abc12
  • 使用默认值

    $ default="hello"; unset var; echo ${var:-$default}
    hello
  • 还有更多

同样,大括号扩展会创建通常在循环中迭代的字符串列表:

$ echo f{oo,ee,a}d
food feed fad

$ mv error.log{,.OLD}
(error.log is renamed to error.log.OLD because the brace expression
expands to "mv error.log error.log.OLD")

$ for num in {000..2}; do echo "$num"; done
000
001
002

$ echo {00..8..2}
00 02 04 06 08

$ echo {D..T..4}
D H L P T

请注意,前导零和增量功能在Bash 4之前不可用。

感谢gboffi提醒我有关括号扩展的信息。

双括号用于算术运算

((a++))

((meaning = 42))

for ((i=0; i<10; i++))

echo $((a + b + (14 * c)))

并且它们使您可以省略整数和数组变量上的美元符号,并在运算符周围包括空格以提高可读性。

单括号也用于数组索引:

array[4]="hello"

element=${array[index]}

右边的(大多数/全部?)数组引用需要大括号。

ephemient的评论提醒我,括号也用于子外壳。并且它们被用来创建数组。

array=(1 2 3)
echo ${array[1]}
2

8
警告:该功能是叉炸弹,请勿运行它。请参阅:en.wikipedia.org/wiki/Fork_bomb
已暂停,直至另行通知。

3
如果您使用额外的调用它,那么它只是一个叉子炸弹:
ephemient

7
同时为了完整,我只是碰到这种排在一个古老的脚本:$[expression]; 这是较新的首选语法的旧的,不赞成使用的算术表达式语法:$((expression))
michael 2012年

2
@DennisWilliamson花括号的另一种用法bash是创建序列,如下文在外围提到的那样(stackoverflow.com/a/8552128/2749397)正如我想对此功能进行一些评论(因为您没有提到它;-)我是m以使用投票最多的答案作为媒介来自由...序列文字的两个示例:echo {01..12}-> 01 02 03 04 05 06 07 08 09 10 11 12(请注意初始的零);echo {C..Q}-> C D E F G H I J K L M N O P Q。它的主要用途是循环for cnt in {01..12} ; do ... ${cnt} ... ; done
播放

1
@gboffi:零填充功能在Bash 4中可用。此外,在Bash 4中,您可以按以下顺序指定增量:echo {01..12..2}->“ 01 03 05 07 09 11”。感谢您提醒有关序列的信息。我将其添加到我的答案中。
暂停,直到另行通知。

335
  1. 一个括号([)通常实际调用一个名为的程序[; man testman [了解更多信息。例:

    $ VARIABLE=abcdef
    $ if [ $VARIABLE == abcdef ] ; then echo yes ; else echo no ; fi
    yes
  2. 双括号([[)与单括号的作用(基本上)相同,但是是内置的bash。

    $ VARIABLE=abcdef
    $ if [[ $VARIABLE == 123456 ]] ; then echo yes ; else echo no ; fi
    no
  3. 括号(())用于创建子外壳。例如:

    $ pwd
    /home/user 
    $ (cd /tmp; pwd)
    /tmp
    $ pwd
    /home/user

    如您所见,子Shell使您可以执行操作而不会影响当前Shell的环境。

  4. (a)大括号({})用于明确标识变量。例:

    $ VARIABLE=abcdef
    $ echo Variable: $VARIABLE
    Variable: abcdef
    $ echo Variable: $VARIABLE123456
    Variable:
    $ echo Variable: ${VARIABLE}123456
    Variable: abcdef123456

    (b)大括号还用于在当前 shell上下文中执行一系列命令,例如

    $ { date; top -b -n1 | head ; } >logfile 
    # 'date' and 'top' output are concatenated, 
    # could be useful sometimes to hunt for a top loader )
    
    $ { date; make 2>&1; date; } | tee logfile
    # now we can calculate the duration of a build from the logfile

与语法有细微的区别( )(参见bash参考); 本质上,;括号中最后一个命令后的分号是必须的,并且括号{} 必须用空格包围。


26
好吧,[实际上是Bash中的内置函数,但它的行为/bin/[[[内置函数相反。 [[具有不同的功能,例如更具逻辑性的操作和不同的引用角色。另外:单个括号也用于数组,进程替换和扩展的glob;双括号用于算术;花括号{}用于命令分组或参数扩展或括号扩展或序列扩展的多种类型。我确定我也错过了其他用途...
短暂

4
表达式中的双等号if [ $VARIABLE == abcdef ]是一种bashism,尽管它很有效,但应该避免;明确使用bash(if [[ ...==...]])或明确使用的是传统的条件(if [ "$VARIABLE" = "abcdef" ])。可以说,脚本应该尽可能简单和可移植,直到它们确实确实需要特定于bash的功能(出于某种原因)。但无论如何,其意图应明确;“ =”和“ ==”和“ [[”和“ [”的工作方式不同,它们的用法应保持一致。
迈克尔2012年

3
@michael_n:为此评论+1。附带一提,我喜欢脚本,但是我发现可移植的方式是通过[ "$var" = ".."]而不是进行测试,这很尴尬==,而在C中,它会分配而不是进行测试(这是导致错误的常见原因)……为什么没有不test使用==代替=?有谁知道?
Olivier Dulac

还有一件有趣的事情是(至少在Kubuntu上)该命令/usr/bin/[不是与的符号链接/usr/bin/test,甚至更多:这些程序甚至有一些不同的大小!
Hi-Angel

另外:单个结束括号)case语句语法的一部分,以结束case行。它没有左括号。这让我第一次看到它就离开了。
奥古斯丁·阿曼纳巴(AgustínAmenabar)'18年

302

括号

if [ CONDITION ]    Test construct  
if [[ CONDITION ]]  Extended test construct  
Array[1]=element1   Array initialization  
[a-z]               Range of characters within a Regular Expression
$[ expression ]     A non-standard & obsolete version of $(( expression )) [1]

[1] http://wiki.bash-hackers.org/scripting/obsolete

大括号

${variable}                             Parameter substitution  
${!variable}                            Indirect variable reference  
{ command1; command2; . . . commandN; } Block of code  
{string1,string2,string3,...}           Brace expansion  
{a..z}                                  Extended brace expansion  
{}                                      Text replacement, after find and xargs

括弧

( command1; command2 )             Command group executed within a subshell  
Array=(element1 element2 element3) Array initialization  
result=$(COMMAND)                  Command substitution, new style  
>(COMMAND)                         Process substitution  
<(COMMAND)                         Process substitution 

双括号

(( var = 78 ))            Integer arithmetic   
var=$(( 20 + 5 ))         Integer arithmetic, with variable assignment   
(( var++ ))               C-style variable increment   
(( var-- ))               C-style variable decrement   
(( var0 = var1<98?9:21 )) C-style ternary operation

@Yola,您能解释一下$ {varname)到底是什么吗?在Apple Xcode项目中,我可以将文件路径指定为脚本输入/输出。如果我指定$ SRC_ROOT / myFile.txt或$ {SRC_ROOT} /myFile.txt(SRC_ROOT var由构建系统导出)-不起作用。仅$(SRC_ROOT)/myFile.txt有效。可能是什么原因?显然var name不是命令吗?
Motti Shneor

1
@MottiShneor,在您的情况下$(varname)与bash语法无关。它是Makefile语法的一部分。
萨沙

并非如此-Xcode不是使用makefile构建的,其变量是环境变量。专有的Xcode构建系统进程读取这些预定义的环境变量的值。定制构建步骤只是普通的shell脚本(bash或其他脚本),并且可以访问相同的var。
Motti Shneor'5

@MottiShneor,好的,让我们完善一下:很可能它是xcconfig语法的一部分。无论如何,$(varname)在您的情况下与bash语法无关。
萨沙

您没有提到测试结构与扩展测试结构之间的区别。
Nikos

23

我只是想从TLDP添加这些:

~:$ echo $SHELL
/bin/bash

~:$ echo ${#SHELL}
9

~:$ ARRAY=(one two three)

~:$ echo ${#ARRAY}
3

~:$ echo ${TEST:-test}
test

~:$ echo $TEST


~:$ export TEST=a_string

~:$ echo ${TEST:-test}
a_string

~:$ echo ${TEST2:-$TEST}
a_string

~:$ echo $TEST2


~:$ echo ${TEST2:=$TEST}
a_string

~:$ echo $TEST2
a_string

~:$ export STRING="thisisaverylongname"

~:$ echo ${STRING:4}
isaverylongname

~:$ echo ${STRING:6:5}
avery

~:$ echo ${ARRAY[*]}
one two one three one four

~:$ echo ${ARRAY[*]#one}
two three four

~:$ echo ${ARRAY[*]#t}
one wo one hree one four

~:$ echo ${ARRAY[*]#t*}
one wo one hree one four

~:$ echo ${ARRAY[*]##t*}
one one one four

~:$ echo $STRING
thisisaverylongname

~:$ echo ${STRING%name}
thisisaverylong

~:$ echo ${STRING/name/string}
thisisaverylongstring

18
介意echo ${#ARRAY}显示三个,因为第一个元素ARRAY包含三个字符,而不是因为它包含三个元素!要打印元素数量,请使用echo ${#ARRAY[@]}
TrueY

@zeal ${TEST:-test}等于$TEST变量是否TEST存在,否则它仅返回字符串“ test”。还有另一个版本可以做更多的事情:${TEST:=test}---也等于$TESTTEST是否存在,但是只要不存在,它就会创建变量TEST并分配值“ test”,并成为整个表达式的值。
Loves Probability

18

BashFAQ中对test[[[之间的区别进行了详细说明。

简而言之:test实现了命令的旧的可移植语法。在几乎所有外壳程序中(最古老的Bourne外壳程序都是例外),[是test的同义词(但需要最后一个参数])。尽管所有现代shell都具有[的内置实现,但是通常仍然存在该名称的外部可执行文件,例如/ bin / [。

[[是它的新改进版本,它是关键字,而不是程序。如下所示,这对易用性具有有益的影响。[[KornShell和BASH(例如2.03)可以理解,但是较旧的POSIX或BourneShell不能理解。

结论:

新的测试命令[[应该在何时使用,旧的[[ 如果需要考虑到BourneShell的可移植性,则应使用旧语法。另一方面,如果脚本需要BASH或KornShell,则新语法更加灵活。


18

函数定义中的括号

()在函数定义中使用 括号:

function_name () { command1 ; command2 ; }

这就是即使在命令参数中也必须转义括号的原因:

$ echo (
bash: syntax error near unexpected token `newline'

$ echo \(
(

$ echo () { command echo The command echo was redefined. ; }
$ echo anything
The command echo was redefined.

哦,我在csh上尝试过。我的错。当我尝试bash时,它可以工作。我不知道bash的命令“命令”。
灿金

如何取消命令echo()的重新定义?(无需重新开始)
Chan Kim

2
@ChanKim: unset -f echo。请参阅help unset
pabouk'2

0
Truncate the contents of a variable

$ var="abcde"; echo ${var%d*}
abc

Make substitutions similar to sed

$ var="abcde"; echo ${var/de/12}
abc12

Use a default value

$ default="hello"; unset var; echo ${var:-$default}
hello

当您回答问题时,不仅要针对“代码”,还要尝试添加解释...
Mikev
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.