it-roy-ru.com

Bash проверка, является ли переменная массивом

У меня есть цикл по именам переменных, и мне нужно проверить, является ли переменная, на которую указывает это имя, массивом или нет. Я пытался гуглить по нему, но ничего не смог найти. Любая помощь?

for varname in AA BB CC; do
  local val
  if [ "$varname" is array ]; then # how can I perform this test?
    echo do something with an array
  else
    echo do something with a "'normal'" variable
  fi
done
27
wujek

Google может быть ваш друг

declare -p variable-name 2> /dev/null | grep -q '^declare \-a'
12
Bastien Jansen

Чтобы избежать вызова grep, вы можете использовать:

if [[ "$(declare -p variable_name)" =~ "declare -a" ]]; then
    echo array
else
    echo no array
fi
29
Reuben W

Начиная с bash 4.3, это уже не так просто.

С помощью команды «объявлять -n» вы можете добавить ссылку на другую переменную и делать это снова и снова. Как будто это не было достаточно сложно, с помощью «объявление -p», вы не получите тип или исходную переменную.

Пример: 

$ declare -a test=( a b c d e)
$ declare -n mytest=test
$ declare -n newtest=mytest
$ declare -p newtest
declare -n newtest="mytest"
$ declare -p mytest
declare -n mytest="test"

Поэтому вы должны просмотреть все ссылки. В bash-only это будет выглядеть так: 

vartype() {
    local var=$( declare -p $1 )
    local reg='^declare -n [^=]+=\"([^\"]+)\"$'
    while [[ $var =~ $reg ]]; do
            var=$( declare -p ${BASH_REMATCH[1]} )
    done

    case "${var#declare -}" in
    a*)
            echo "ARRAY"
            ;;
    A*)
            echo "HASH"
            ;;
    i*)
            echo "INT"
            ;;
    x*)
            echo "EXPORT"
            ;;
    *)
            echo "OTHER"
            ;;
    esac
}

С приведенным выше примером:

$ vartype newtest
ARRAY

Чтобы проверить массив, вы можете изменить код или использовать его с grep: 

vartype $varname | grep -q "ARRAY"
7
Marco

Я начал с Рувима отличный ответ выше. Я реализовал несколько комментариев и некоторые из моих собственных улучшений и получил следующее:

#!/bin/bash
array_test() {
    # no argument passed
    [[ $# -ne 1 ]] && echo 'Supply a variable name as an argument'>&2 && return 2
    var=$1
    # use a variable to avoid having to escape spaces
    regex="^declare -[aA] ${var}(=|$)"
    [[ $(declare -p "$var" 2> /dev/null) =~ $regex ]] && return 0
}

Теперь я могу сделать это:

foo=(lorem ipsum dolor)
bar="declare -a tricky"
declare -A baz

array_test foo && echo "it's an array"
array_test bar && echo "it's an array"
# properly detects empty arrays
array_test baz && echo "it's an array"
# won't throw errors on undeclared variables
array_test foobarbaz && echo "it's an array"
0
miken32
is_array() {
  local variable_name=$1
  [[ "$(declare -p $variable_name)" =~ "declare -a" ]]
}

is_array BASH_VERSINFO && echo BASH_VERSINFO is an array

is_array() {
    local variable_name=$1
    [[ "$(declare -p $variable_name 2>/dev/null)" =~ "declare -a" ]]
}
0
dawesh