Set*_*jmp 149 string bash shell scripting null
如果我想检查空字符串,我会这样做
[ -z $mystr ]
Run Code Online (Sandbox Code Playgroud)
但是,如果我想检查变量是否已定义呢?或者bash脚本没有区别?
Jon*_*ler 136
我认为你所追求的答案是由Vinko的回答所暗示的(如果没有说明的话),尽管它没有简单说明.要区分VAR是否已设置但为空或未设置,您可以使用:
if [ -z "${VAR+xxx}" ]; then echo VAR is not set at all; fi
if [ -z "$VAR" ] && [ "${VAR+xxx}" = "xxx" ]; then echo VAR is set but empty; fi
Run Code Online (Sandbox Code Playgroud)
您可以将第二行的两个测试合并为一个:
if [ -z "$VAR" -a "${VAR+xxx}" = "xxx" ]; then echo VAR is set but empty; fi
Run Code Online (Sandbox Code Playgroud)
但是,如果您阅读Autoconf的文档,您会发现他们不建议将术语与' -a' 结合使用,并建议使用单独的简单测试&&.我没遇到过有问题的系统; 这并不意味着它们过去不存在(但它们现在可能极为罕见,即使它们在遥远的过去并不罕见).
您可以在Bash手册中找到这些的详细信息,以及其他相关的shell参数扩展,test或[命令和条件表达式.
我最近通过电子邮件询问了这个问题:
你使用两个测试,我理解第二个,但不是第一个.更确切地说,我不了解变量扩展的必要性
Run Code Online (Sandbox Code Playgroud)if [ -z "${VAR+xxx}" ]; then echo VAR is not set at all; fi这不会完成同样的事吗?
Run Code Online (Sandbox Code Playgroud)if [ -z "${VAR}" ]; then echo VAR is not set at all; fi
公平的问题 - 答案是'不,你更简单的选择不会做同样的事情'.
假设我在测试之前写这个:
VAR=
Run Code Online (Sandbox Code Playgroud)
你的测试会说"VAR根本没有设置",但是我会说(暗示因为它没有回应)"VAR已设置但其值可能为空".试试这个脚本:
(
unset VAR
if [ -z "${VAR+xxx}" ]; then echo JL:1 VAR is not set at all; fi
if [ -z "${VAR}" ]; then echo MP:1 VAR is not set at all; fi
VAR=
if [ -z "${VAR+xxx}" ]; then echo JL:2 VAR is not set at all; fi
if [ -z "${VAR}" ]; then echo MP:2 VAR is not set at all; fi
)
Run Code Online (Sandbox Code Playgroud)
输出是:
JL:1 VAR is not set at all
MP:1 VAR is not set at all
MP:2 VAR is not set at all
Run Code Online (Sandbox Code Playgroud)
在第二对测试中,设置变量,但将其设置为空值.这是${VAR=value}和${VAR:=value}符号的区别.同上,用于${VAR-value}和${VAR:-value},和${VAR+value}和${VAR:+value},等等.
正如Gili在他的回答中指出的那样,如果你bash使用该set -o nounset选项,那么上面的基本答案就会失败unbound variable.它很容易补救:
if [ -z "${VAR+xxx}" ]; then echo VAR is not set at all; fi
if [ -z "${VAR-}" ] && [ "${VAR+xxx}" = "xxx" ]; then echo VAR is set but empty; fi
Run Code Online (Sandbox Code Playgroud)
或者您可以取消set -o nounset选项set +u(set -u相当于set -o nounset).
Vin*_*vic 37
~> if [ -z $FOO ]; then echo "EMPTY"; fi
EMPTY
~> FOO=""
~> if [ -z $FOO ]; then echo "EMPTY"; fi
EMPTY
~> FOO="a"
~> if [ -z $FOO ]; then echo "EMPTY"; fi
~>
Run Code Online (Sandbox Code Playgroud)
-z也适用于未定义的变量.未定义和区分定义的,你会用列出的东西在这里或者,更清晰的解释,在这里.
最干净的方式是使用这些示例中的扩展.要获得所有选项,请检查手册的"参数扩展"部分.
替代词:
~$ unset FOO
~$ if test ${FOO+defined}; then echo "DEFINED"; fi
~$ FOO=""
~$ if test ${FOO+defined}; then echo "DEFINED"; fi
DEFINED
Run Code Online (Sandbox Code Playgroud)
默认值:
~$ FOO=""
~$ if test "${FOO-default value}" ; then echo "UNDEFINED"; fi
~$ unset FOO
~$ if test "${FOO-default value}" ; then echo "UNDEFINED"; fi
UNDEFINED
Run Code Online (Sandbox Code Playgroud)
当然,您可以使用其中一种方法,将所需值设置为"默认值"并直接使用扩展(如果适用).
小智 18
高级bash脚本编写指南,10.2.参数替换:
根据是否定义了变量$ mystr来编写程序逻辑,可以执行以下操作:
isdefined=0
${mystr+ export isdefined=1}
Run Code Online (Sandbox Code Playgroud)
现在,如果isdefined = 0则变量未定义,如果isdefined = 1,则定义变量
这种检查变量的方法比上面的答案要好,因为它更优雅,可读,并且如果你的bash shell配置为使用未定义的变量时出错(set -u),脚本将过早终止.
其他有用的东西:
如果未定义,则将默认值7分配给$ mystr,否则保持原样:
mystr=${mystr- 7}
Run Code Online (Sandbox Code Playgroud)
如果变量未定义,则打印错误消息并退出函数:
: ${mystr? not defined}
Run Code Online (Sandbox Code Playgroud)
请注意,我使用':'以便在定义时不将$ mystr的内容作为命令执行.
测试摘要.
[ -n "$var" ] && echo "var is set and not empty"
[ -z "$var" ] && echo "var is unset or empty"
[ "${var+x}" = "x" ] && echo "var is set" # may or may not be empty
[ -n "${var+x}" ] && echo "var is set" # may or may not be empty
[ -z "${var+x}" ] && echo "var is unset"
[ -z "${var-x}" ] && echo "var is set and empty"
Run Code Online (Sandbox Code Playgroud)
检查正在定义的变量的显式方法是:
[ -v mystr ]
Run Code Online (Sandbox Code Playgroud)
/sf/answers/687746041/包含一个更好的答案(一个更具可读性且适用于set -o nounset启用的答案).它的工作方式大致如下:
if [ -n "${VAR-}" ]; then
echo "VAR is set and is not empty"
elif [ "${VAR+DEFINED_BUT_EMPTY}" = "DEFINED_BUT_EMPTY" ]; then
echo "VAR is set, but empty"
else
echo "VAR is not set"
fi
Run Code Online (Sandbox Code Playgroud)