Bash
Когда использовать eval
Поиск…
Вступление
Прежде всего, знайте, что вы делаете! Во-вторых, в то время как вам следует избегать использования 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
.