web-dev-qa-db-ja.com

bashで変数をインクリメントする方法は?

var=$var+1var=($var+1)の両方を使用して数値変数をインクリメントしようとしましたが、成功しませんでした。変数は数値ですが、bashは文字列として読み取っているように見えます。

Ubuntu 13.10上のBashバージョン4.2.45(1)-release(x86_64-pc-linux-gnu).

597
user221744

Bashで変数をインクリメントする方法は複数ありますが、試したことは正しくありません。

たとえば、算術展開を使用できます。

var=$((var+1))
((var=var+1))
((var+=1))
((var++))

または、letを使用できます。

let "var=var+1"
let "var+=1"
let "var++"

http://tldp.org/LDP/abs/html/dblparens.html も参照してください。

926
Radu Rădeanu
var=$((var + 1))

Bashの算術演算は$((...))構文を使用します。

154
Paul Tanzini

さまざまなオプションのパフォーマンス分析

RaduRădeanu's answer のおかげで、bashで変数をインクリメントする次の方法が提供されます。

var=$((var+1))
((var=var+1))
((var+=1))
((var++))
let "var=var+1"
let "var+=1" 
let "var++"

他の方法もあります。たとえば、この質問に関する他の回答をご覧ください。

let var++
var=$((var++))
((++var))
{
    declare -i var
    var=var+1
    var+=1
}
{
    i=0
    i=$(expr $i + 1)
}

非常に多くのオプションがあると、次の2つの質問につながります。

  1. それらの間にパフォーマンスの違いはありますか?
  2. もしそうなら、どれが最高のパフォーマンスを発揮しますか?

増分パフォーマンステストコード:

#!/bin/bash

# To focus exclusively on the performance of each type of increment
# statement, we should exclude bash performing while loops from the
# performance measure. So, let's time individual scripts that
# increment $i in their own unique way.

# Declare i as an integer for tests 12 and 13.
echo > t12 'declare -i i; i=i+1'
echo > t13 'declare -i i; i+=1'
# Set i for test 14.
echo > t14 'i=0; i=$(expr $i + 1)'

x=100000
while ((x--)); do
    echo >> t0 'i=$((i+1))'
    echo >> t1 'i=$((i++))'
    echo >> t2 '((i=i+1))'
    echo >> t3 '((i+=1))'
    echo >> t4 '((i++))'
    echo >> t5 '((++i))'
    echo >> t6 'let "i=i+1"'
    echo >> t7 'let "i+=1"'
    echo >> t8 'let "i++"'
    echo >> t9 'let i=i+1'
    echo >> t10 'let i+=1'
    echo >> t11 'let i++'
    echo >> t12 'i=i+1'
    echo >> t13 'i+=1'
    echo >> t14 'i=$(expr $i + 1)'
done

for script in t0 t1 t2 t3 t4 t5 t6 t7 t8 t9 t10 t11 t12 t13 t14; do
    line1="$(head -1 "$script")"
    printf "%-24s" "$line1"
    { time bash "$script"; } |& grep user
    # Since stderr is being piped to grep above, this will confirm
    # there are no errors from running the command:
    eval "$line1"
    rm "$script"
done

結果:

i=$((i+1))              user    0m0.992s
i=$((i++))              user    0m0.964s
((i=i+1))               user    0m0.760s
((i+=1))                user    0m0.700s
((i++))                 user    0m0.644s
((++i))                 user    0m0.556s
let "i=i+1"             user    0m1.116s
let "i+=1"              user    0m1.100s
let "i++"               user    0m1.008s
let i=i+1               user    0m0.952s
let i+=1                user    0m1.040s
let i++                 user    0m0.820s
declare -i i; i=i+1     user    0m0.528s
declare -i i; i+=1      user    0m0.492s
i=0; i=$(expr $i + 1)   user    0m5.464s

結論:

i+=1が整数として宣言されている場合、bashは$iの実行が最も速いようです。 letステートメントは特に遅いように見えますが、exprは組み込みではないため、最も遅いです。

81
Keith Reynolds

これもあります:

var=`expr $var + 1`

スペースに注意してください。また、`'ではありません

Raduの回答とコメントは網羅的で非常に有用ですが、bash固有のものです。 bashについて具体的に質問したことは知っていますが、uCLinuxのbusyboxでshを使用して同じことをしようとしていたときにこの質問を見つけたので、パイプを入れようと思いました。このポータブルはbashを超えています。

18
tphelican

$varを整数として宣言すると、最初に試したことが実際に機能します。

$ declare -i var=5
$ echo $var
5
$ var=$var+1
$ echo $var
6

リファレンス: 変数の種類、初心者向けBashガイド

10

すべての回答に1つのメソッドがありません-bc

$ VAR=7    
$ bc <<< "$VAR+2"
9
$ echo $VAR
7
$ VAR=$( bc <<< "$VAR+1" )
$ echo $VAR
8

bcPOSIX standardで指定されるため、UbuntuおよびPOSIX準拠システムのすべてのバージョンに存在する必要があります。 <<<リダイレクトは、移植性のためにecho "$VAR" | bcに変更できますが、質問ではbashについて尋ねられるため、<<<を使用するだけでかまいません。

7

リターンコード1の問題は、すべてのデフォルトバリアント(let(())など)に存在します。これは、たとえば、set -o errexitを使用するスクリプトで問題を引き起こすことがあります。以下は、1と評価される数式からのエラーコード0を防ぐために使用しているものです。

math() { (( "$@" )) || true; }

math a = 10, b = 10
math a++, b+=2
math c = a + b
math mod = c % 20
echo $a $b $c $mod
#11 12 23 3
6
Juve

がらくた、これは別のディスカッションスレッド用でした-投稿を削除する方法がわからない

#!/bin/bash
myname=$(basename "$0")

# parse sub options
get_opts () {
  rs='' && rc=0 # return string and return code
  while [[ $# -gt 0 ]]; do
    shift
    [[ "$1" =~ -.* ]] && break ||  rs="$rs $1" && rc=$((rc + 1))
  done
  echo "$rs"
}

# help function
help () { cat <<EOP
   $myname: -c cluster [...] -a action [...] -i instance [...]
EOP
}

while [[ $# -gt 0 ]]; do
    case $1 in
        "-a") ACTS="$(get_opts $@)"
           ;;
        "-i") INSTS=$(get_opts $@)
           ;;
        "-c") CLUSTERS=$(get_opts $@)
           ;;
        "-h") help
           ;;
        ?) echo "sorry, I dont do $1"
           exit
           ;;
    esac
    shift
done

exit
0
Neil Verkland