Ответ 1
Редактировать: это более полная версия, которая показывает больше различий между [
(иначе test
) и [[
.
В следующей таблице показано, что независимо от того, указана ли переменная в кавычках или нет, используете ли вы одинарные или двойные скобки и содержит ли переменная только пробел, это то, что влияет на то, подходит ли для проверки использование теста с -n / -z
Переменная.
| 1a 2a 3a 4a 5a 6a | 1b 2b 3b 4b 5b 6b
| [ [" [-n [-n" [-z [-z" | [[ [[" [[-n [[-n" [[-z [[-z"
-----+------------------------------------+------------------------------------
unset| false false true false true true | false false false false true true
null | false false true false true true | false false false false true true
space| false true true true true false| true true true true false false
zero | true true true true false false| true true true true false false
digit| true true true true false false| true true true true false false
char | true true true true false false| true true true true false false
hyphn| true true true true false false| true true true true false false
two | -err- true -err- true -err- false| true true true true false false
part | -err- true -err- true -err- false| true true true true false false
Tstr | true true -err- true -err- false| true true true true false false
Fsym | false true -err- true -err- false| true true true true false false
T= | true true -err- true -err- false| true true true true false false
F= | false true -err- true -err- false| true true true true false false
T!= | true true -err- true -err- false| true true true true false false
F!= | false true -err- true -err- false| true true true true false false
Teq | true true -err- true -err- false| true true true true false false
Feq | false true -err- true -err- false| true true true true false false
Tne | true true -err- true -err- false| true true true true false false
Fne | false true -err- true -err- false| true true true true false false
Если вы хотите узнать, имеет ли переменная ненулевую длину, выполните одно из следующих действий:
- заключить переменную в квадратные скобки (столбец 2a)
- используйте
-n
и-n
кавычки переменную (столбец 4a) - используйте двойные скобки с или без
-n
и с или без-n
(столбцы 1b - 4b)
Обратите внимание, что в столбце 1a, начинающемся со строки, помеченной как "два", результат указывает, что [
оценивает содержимое переменной так, как если бы они были частью условного выражения (результат соответствует утверждению, подразумеваемому "T" или "F") в столбце описания). Когда используется [[
(столбец 1b)), содержимое переменной рассматривается как строка и не оценивается.
Ошибки в столбцах 3a и 5a вызваны тем фактом, что значение переменной содержит пробел, а переменная не заключена в кавычки. Опять же, как показано в столбцах 3b и 5b, [[
оценивает содержимое переменной как строку.
Соответственно, для тестов для строк нулевой длины столбцы 6a, 5b и 6b показывают правильные способы сделать это. Также обратите внимание, что любой из этих тестов может быть отменен, если отрицание показывает более ясное намерение, чем использование противоположной операции. Например: if ! [[ -n $var ]]
if ! [[ -n $var ]]
.
Если вы используете [
, ключом к тому, чтобы вы не получили неожиданных результатов, является кавычка переменной. Использование [[
, это не имеет значения.
Сообщения об ошибках, которые подавляются, представляют собой "ожидается унарный оператор" или "ожидается двоичный оператор".
Это скрипт, который создал таблицу выше.
#!/bin/bash
# by Dennis Williamson
# 2010-10-06, revised 2010-11-10
# for http://stackoverflow.com/q/3869072
# designed to fit an 80 character terminal
dw=5 # description column width
w=6 # table column width
t () { printf '%-*s' "$w" " true"; }
f () { [[ $? == 1 ]] && printf '%-*s' "$w" " false" || printf '%-*s' "$w" " -err-"; }
o=/dev/null
echo ' | 1a 2a 3a 4a 5a 6a | 1b 2b 3b 4b 5b 6b'
echo ' | [ [" [-n [-n" [-z [-z" | [[ [[" [[-n [[-n" [[-z [[-z"'
echo '-----+------------------------------------+------------------------------------'
while read -r d t
do
printf '%-*s|' "$dw" "$d"
case $d in
unset) unset t ;;
space) t=' ' ;;
esac
[ $t ] 2>$o && t || f
[ "$t" ] && t || f
[ -n $t ] 2>$o && t || f
[ -n "$t" ] && t || f
[ -z $t ] 2>$o && t || f
[ -z "$t" ] && t || f
echo -n "|"
[[ $t ]] && t || f
[[ "$t" ]] && t || f
[[ -n $t ]] && t || f
[[ -n "$t" ]] && t || f
[[ -z $t ]] && t || f
[[ -z "$t" ]] && t || f
echo
done <<'EOF'
unset
null
space
zero 0
digit 1
char c
hyphn -z
two a b
part a -a
Tstr -n a
Fsym -h .
T= 1 = 1
F= 1 = 2
T!= 1 != 2
F!= 1 != 1
Teq 1 -eq 1
Feq 1 -eq 2
Tne 1 -ne 2
Fne 1 -ne 1
EOF