したがって、次のように間接拡張によって一般パラメータの有無をテストできることがわかります。
foo=1
bar=foo
(( ${(P)+bar} )) && print "$bar exists"
以下を実行して、連想配列にキーが存在するかどうかをテストできることを知っています。
foo=([abc]=1)
(( ${+foo[abc]} )) && print "abc exists"
しかし、2つを組み合わせて、間接拡張によってキーが連想配列に存在するかどうかをテストする方法はわかりません。 evalを使用しないことは可能ですか?
以下を含むいくつかの組み合わせを試しましたが、そのうち何も機能しません。
foo=([abc]=1)
bar=foo
(( ${(P)+bar[abc]} )) && print "$bar has key abc" # Test fails
(( ${(P)+${bar}[abc]} )) && print "$bar has key abc" # Passes for nonexistant keys
(( ${${(P)+bar}[abc]} )) && print "$bar has key abc" # Test fails
(( ${${(P)bar}+[abc]} )) && print "$bar has key abc" # prints "zsh: bad output format specification"
答え1
いいえここで議論してください。値の渡しを防ぐには、${:-word}
他のパラメータ拡張(たとえば)で正しい形式の文字列を使用してから、次のように拡張する必要があります${(P)+...}
。
(( ${(P)+${:-${bar}[abc]}} )) && print OK || print FAIL
答え2
の使用方法はわかりませんが、${+param}
次のことができます[[ -v $param ]]
。
foo=([abc]=1)
bar=foo
[[ -v "$bar""[abc]" ]] && print "$bar has key abc"
# or "$bar"[abc] or $bar''[abc] or $bar'[abc]'
# or any other way to suppress $bar[abc] being interpreted as a value in $bar
答え3
私は式を書いてevalを使うことがトリックであることがわかりました。
typeset -A ARRAY_TEST
ARRAY_TEST[key1]=
ARRAY_TEST[key2]=value2
ARRAY_NAME=ARRAY_TEST
# Note: ${NAME+1} will return 1 if a variable is set, otherwise nothing
# for example, ${ARRAY_TEST[key1]+1} will be 1 and
# but ${ARRAY_TEST[key3]+1} will be nothing
[[ $(eval "echo \${${ARRAY_NAME}[key1]+1}") == 1 ]] && echo has key1 || echo no key1
[[ $(eval "echo \${${ARRAY_NAME}[key2]+1}") == 1 ]] && echo has key2 || echo no key2
[[ $(eval "echo \${${ARRAY_NAME}[key3]+1}") == 1 ]] && echo has key3 || echo no key3
# Outputs:
#
# has key1
# has key2
# no key3
結局これのための関数を作りました。
function array-has-value() {
local testVariable=$1
local keyValue=$2
[[ $(eval "echo \${${testVariable}[$keyValue]+1}") == 1 ]] && return 0 || return 1
}
以下は上記と同じ結果を生成します。
array-has-value ARRAY_TEST key1 && echo has key1 || echo no key1
array-has-value ARRAY_TEST key2 && echo has key2 || echo no key2
array-has-value ARRAY_TEST key3 && echo has key3 || echo no key3
答え4
私が見つけたもう一つの答えは次のとおりです。
typeset -A foo=([abc]=def)
has_key() {
local var="${1}[$2]"
(( ${(P)+${var}} )) && return 0
return 1
}
has_key foo abc && print "foo has abc"
has_key foo def || print "foo doesn't have def"
# Outputs:
# foo has abc
# foo doesn't have def