Поиск…


Вступление

Прежде всего, знайте, что вы делаете! Во-вторых, в то время как вам следует избегать использования eval , если его использование делает более чистый код, продолжайте.

Использование Eval

Например, рассмотрим следующее, которое задает содержимое $@ содержимому данной переменной:

a=(1 2 3)
eval set -- "${a[@]}"

Этот код часто сопровождается getopt или getopts чтобы установить $@ на вывод вышеупомянутых парсеров параметров, однако вы также можете использовать его для создания простой функции pop которая может работать с переменными молча и напрямую, не сохраняя результат исходная переменная:

isnum()
{
    # is argument an integer?
    local re='^[0-9]+$'
    if [[ -n $1 ]]; then
        [[ $1 =~ $re ]] && return 0
        return 1
    else
        return 2
    fi
}

isvar()
{
    if isnum "$1"; then
        return 1
    fi
    local arr="$(eval eval -- echo -n "\$$1")"
    if [[ -n ${arr[@]} ]]; then
        return 0
    fi
    return 1
}

pop()
{
    if [[ -z $@ ]]; then
        return 1
    fi

    local var=
    local isvar=0
    local arr=()
    
    if isvar "$1"; then # let's check to see if this is a variable or just a bare array
        var="$1"
        isvar=1
        arr=($(eval eval -- echo -n "\${$1[@]}")) # if it is a var, get its contents
    else
        arr=($@)
    fi
    
    # we need to reverse the contents of $@ so that we can shift
    # the last element into nothingness
    arr=($(awk <<<"${arr[@]}" '{ for (i=NF; i>1; --i) printf("%s ",$i); print $1; }'

    # set $@ to ${arr[@]} so that we can run shift against it.
    eval set -- "${arr[@]}"
    
    shift # remove the last element
    
    # put the array back to its original order
    arr=($(awk <<<"$@" '{ for (i=NF; i>1; --i) printf("%s ",$i); print $1; }'
    
    # echo the contents for the benefit of users and for bare arrays
    echo "${arr[@]}"

    if ((isvar)); then
        # set the contents of the original var to the new modified array
        eval -- "$var=(${arr[@]})" 
    fi
}

Использование Eval с Getopt

В то время как eval может не понадобиться для функции pop , это необходимо, когда вы используете getopt :

Рассмотрим следующую функцию, которая принимает -h как вариант:

f()
{
    local __me__="${FUNCNAME[0]}"
    local argv="$(getopt -o 'h' -n $__me__ -- "$@")"

    eval set -- "$argv"
    
    while :; do
        case "$1" in
            -h)
              echo "LOLOLOLOL"
              return 0
              ;;
            --)
              shift
              break
              ;;
    done
    
    echo "$@"
}

Без eval set -- "$argv" генерирует -h -- вместо желаемого (-h --) и затем вводит бесконечный цикл, потому что -h -- не соответствует -- или -h .



Modified text is an extract of the original Stack Overflow Documentation
Лицензировано согласно CC BY-SA 3.0
Не связан с Stack Overflow