Как использовать двойные или одинарные скобки, скобки, фигурные скобки

581

Меня смущает использование скобок, круглых скобок, фигурных скобок в Bash, а также разница между их двойными или одиночными формами. Есть ли ясное объяснение?

Теги:
syntax

6 ответов

508
Лучший ответ

В Bash, test и [ встроены.

двойная скобка обеспечивает дополнительную функциональность. Например, вы можете использовать && и || вместо -a и -o и там оператор регулярного выражения =~.

Скобки, в дополнение к разграничению имени переменной, используются для расширения параметров, чтобы вы могли делать такие вещи, как:

  • Обрезать содержимое переменной

    $ 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)))

и они позволяют вам опускать знаки доллара на переменные integer и array и включать пробелы вокруг операторов для удобочитаемости.

Отдельные скобки также используются для массива:

array[4]="hello"

element=${array[index]}

Вставные скобки необходимы для ссылок массива (большинство/все?) с правой стороны.

Комментарий

ephemient напомнил мне, что круглые скобки также используются для подоболочек. И они используются для создания массивов.

array=(1 2 3)
echo ${array[1]}
2
  • 7
    ВНИМАНИЕ: эта функция является бомбой-вилкой, не запускайте ее. Смотрите: en.wikipedia.org/wiki/Fork_bomb
  • 3
    Это только вилочная бомба, если вы вызываете ее с дополнительным :
Показать ещё 5 комментариев
272
  • Одна скобка ([) обычно на самом деле вызывает программу с именем [; man test или man [ для получения дополнительной информации. Пример:

    $ VARIABLE=abcdef
    $ if [ $VARIABLE == abcdef ] ; then echo yes ; else echo no ; fi
    yes
    
  • Двойная скобка ([[) делает одно и то же (в основном) как единую скобку, но является встроенным bash.

    $ VARIABLE=abcdef
    $ if [[ $VARIABLE == 123456 ]] ; then echo yes ; else echo no ; fi
    no
    
  • Круглые скобки (()) используются для создания подоболочки. Например:

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

    Как вы можете видеть, подоболочка позволяла выполнять операции, не затрагивая среду текущей оболочки.

4а. Скобки ({}) используются для однозначной идентификации переменных. Пример:

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

4b. Скобки также используются для выполнения последовательности команд в текущем контексте оболочки, например.

    $ { 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 ссылка); по существу, точка с запятой ; после того, как последняя команда в фигурных скобках является обязательной, а фигурные скобки {, } должны быть окружены пробелами.

  • 25
    Ну, на самом деле [ это встроенная функция в Bash, но она должна действовать как /bin/[ а не как [[ builtin. [[ имеет различные функции, такие как более логические операции и разные роли цитирования. Дополнительно: одиночные скобки также используются для массивов, подстановки процессов и расширенных глобанов; двойные скобки используются для арифметики; фигурные скобки {} используются для группировки команд или множества типов расширения параметров или расширения фигурных скобок или расширения последовательности. Я уверен, что я пропустил и некоторые другие применения ...
  • 4
    Двойное равенство в выражении, if [ $VARIABLE == abcdef ] является башизмом, который - хотя он работает - вероятно, следует избегать; либо явно используйте bash ( if [[ ...==...]] ), либо дайте понять, что вы используете более традиционные условные выражения ( if [ "$VARIABLE" = "abcdef" ] ). Можно утверждать, что сценарии должны начинаться как можно более простыми и переносимыми до тех пор, пока им действительно не понадобятся функции, специфичные для bash (по той или иной причине). Но в любом случае намерение должно быть ясным; «=» и «==» и «[[» и «[» работают по-разному, и их использование должно быть согласованным.
Показать ещё 3 комментария
247

Скобки

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
  • 0
    @Yola, не могли бы вы объяснить, что именно означает $ (varname)? В проектах Apple Xcode я могу указать пути к файлам в качестве сценария ввода / вывода. если я укажу $ SRC_ROOT / myFile.txt или $ {SRC_ROOT} /myFile.txt (SRC_ROOT var экспортируется системой сборки) - не работает. работает только $ (SRC_ROOT) /myFile.txt. Что может быть причиной? ясно, что имя var не команда?
  • 1
    @MottiShneor, в вашем случае $(varname) имеет отношения к синтаксису bash. Это часть синтаксиса Makefile .
Показать ещё 3 комментария
21

Я просто хотел добавить их из 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
  • 16
    Имейте в виду, что echo ${#ARRAY} отображает три, потому что первый элемент ARRAY содержит три символа, а не потому, что он содержит три элемента! Чтобы напечатать количество элементов, используйте echo ${#ARRAY[@]} .
  • 0
    @zeal ${TEST:-test} равно $TEST если переменная TEST существует, в противном случае она просто возвращает строку «test». Есть еще одна версия, которая делает еще больше: ${TEST:=test} --- которая также равна $TEST если TEST существует, но всякий раз, когда ее нет, она создает переменную TEST и присваивает значение «test», а также становится значением всего выражения.
17

Разница между тестом, [ и [[] подробно объясняется в BashFAQ.

Сокращение длинной истории: тест реализует старый, переносимый синтаксис команда. Почти во всех раковинах (самые старые ракеты Борна - это исключение), [является синонимом теста (но требует окончательного аргумента ]). Хотя все современные оболочки имеют встроенные реализации [, обычно остается внешний исполняемый файл этого имени, например. /Бен/[.

[[это новая улучшенная версия, которая является ключевым словом, а не программой). Это благотворно влияет на простоту использования, как показано ниже. [[ является понимается KornShell и BASH (например, 2.03), но не более старыми POSIX или BourneShell.

И заключение:

Когда должна появиться новая тестовая команда [[будет использоваться, а когда старая [? Если переносимость BourneShell вызывает беспокойство, старый синтаксис должен использоваться. Если, с другой стороны, для script требуется BASH или KornShell, новый синтаксис гораздо более гибкий.

16

Скобки в определении функции

Круглые скобки () используются в определении функции:

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.
  • 0
    О, я попробовал на csh. Виноват. Когда я примеряю bash, это работает. Я не знал команду «команда» Баш.
  • 0
    как я могу отменить переопределение команды echo ()? (без повторного открытия Bash)
Показать ещё 1 комментарий

Ещё вопросы

Сообщество Overcoder
Наверх
Меню