질문에 대한 간략한 설명 :
이 내장되어 배시에있어서, 상기 어레이의 이름 (즉 변수에 저장) 동적 배시 배열의 요소 수를 계산하기 위해, 하지 A는 완전히 배열의 카피 제조 또는 사용에 의존 eval
?
추가 정보:
bash 매개 변수 대체를 사용하여 다음을 수행 할 수 있습니다.
- 배열의 길이를 결정하십시오
myArr=(A B C); echo ${#myArr[@]}
. - 이름으로 변수를 간접적으로 참조하십시오 :
NAME=myVar; echo ${!NAME}
(배열 요소에도 적용됩니다) :
NAME=myArr[1]; echo ${!NAME}
그러나 배열의 이름이 다른 변수에 저장되어 있으면 배열의 요소 수를 어떻게 결정할 수 있습니까? ( 이를 위의 두 매개 변수 대체 조합 으로 고려할 수 있습니다 .) 예를 들면 다음과 같습니다.
myArr=(A B C D)
NAME=myArr
# Get the number of elements in the array indirectly referenced by NAME.
count=${#$NAME[@]} # This syntax is invalid. What is the right way?
다음은 모두 실패한 여러 시도입니다.
# Setup for following attempts:
myArr=(A B C D)
NAME=myArr
EXPR1=$NAME[@] # i.e. EXPR1='myArr[@]'
EXPR2=#$NAME[@] # i.e. EXPR2='#myArr[@]'
# Failed attempts to get the lengh of the array indirectly:
1. count=${#$NAME[@]} # ERROR: bash: ...: bad substitution
2. count=${#!EXPR1} # ERROR: bash: !EXPR}: event not found
3. count=${#\!EXPR1} # ERROR: bash: ...: bad substitution
4. count=${!#EXPR1} # ERROR: bash: ...: bad substitution
5. count=${!EXPR2} # Returns NULL
또한 위의 다른 변형을 시도했지만 (A) 배열의 사본을 만들거나 (B)를 사용하여 작동하지 않는 것을 아직 찾지 못했습니다. eval
.
작업 방법 :
이 문제를 해결하는 두 가지 방법이 최적이 아닐 수도 있습니다 (그러나 내가 틀렸다면 정정하십시오).
방법 1 : 배열 복사
배열을 다른 (정적으로 명명 된) 변수에 할당하고 그 안의 요소 수를 가져옵니다.
EXPR=$NAME[@]
arrCopy=( "${!EXPR}" )
count=${#arrCopy}
방법 2 : 사용 eval
EXPR="count=\${#$NAME[@]}" # i.e. 'count=${myArr[@]}'
eval $EXPR
# Now count is set to the length of the array
요약:
bash에 배열의 길이를 간접적으로 결정하는 내장 메소드 (예 : 매개 변수 대체 구문)가 있습니까? 그렇지 않은 경우 가장 효율적인 방법은 무엇입니까? eval
위 의 방법 이라고 가정 하지만 보안 또는 성능 문제가 eval
있습니까?
답변
당신은 인덱스 에바에서 그 것들을 처리해야합니다. 배열 변수로 만들면 간접 변수의 색인을 통해 간접적으로 할 수 있습니다 .
a=(abc1 def2 ghi3 jkl4 mno5)
r=('a[c=${#a[@]}]' a\[i] a\[@])
for i in 0 1 2 3 4 5
do c=
printf "<%s>\n" "${!r-${!r[i<c?1:2]}}"
printf "\n\tindex is $i and count is $c\n\n"
done
<abc1>
index is 0 and count is 5
<def2>
index is 1 and count is 5
<ghi3>
index is 2 and count is 5
<jkl4>
index is 3 and count is 5
<mno5>
index is 4 and count is 5
<abc1>
<def2>
<ghi3>
<jkl4>
<mno5>
index is 5 and count is 5
bash
의 인덱스는 0을 기준으로 하기 때문에 배열 객체의 총 개수는 항상 가장 높은 세트 인덱스보다 하나 이상으로 계산됩니다.
c=
echo "${a[c=${#a[@]}]-this index is unset}" "$c"
this index is unset 5
… 제공된 경우이 매개 변수는 기본 단어로 확장됩니다.
제공되지 않은 경우 :
c=
${!r}
echo "$c"
5
… 해를 끼치 지 않습니다.
루프에서 나는 $i
ndex 변수를 추적하고 적어도 $c
ount 만큼 큰지 확인합니다 . 더 적을 때 유효한 인덱스이기 때문에 $r
evar var를 확장 a[i]
하지만, 같거나 크면 $r
ef를 전체 $a
rray 로 확장합니다 .
다음은 함수입니다.
ref_arr(){
local index=-1 count=
local ref=( "$1[ count= \${#$1[@]} ]"
"$1[ index ]" "$1[ @ ]"
) && printf "input array '%s' has '%d' members.\n" \
"$1" "${!ref-${count:?invalid array name: "'$1'"}}"
while [ "$((index+=1))" -lt "$count" ]
do printf "$1[$index] == '%s'\n" "${!ref[1]}"
done
}
some_array=(some "dumb
stuff" 12345\'67890 "" \
'$(kill my computer)')
ref_arr some_array
ref_arr '$(echo won'\''t work)'
input array 'some_array' has '5' members.
some_array[0] == 'some'
some_array[1] == 'dumb
stuff'
some_array[2] == '12345'67890'
some_array[3] == ''
some_array[4] == '$(kill my computer)'
bash: count: invalid array name: '$(echo won't work)'
답변
bash 4.3 namerefs는 신의 선물입니다. 그러나 다음을 수행 할 수 있습니다.
$ myArr=(A B C D)
$ NAME=myArr
$ tmp="${NAME}[@]"
$ copy=( "${!tmp}" )
$ echo "${#copy[@]}"
4