コンテンツにスキップ

二進対数

出典: フリー百科事典『ウィキペディア(Wikipedia)』
log2 nグラフ

二進対数 (にしんたいすう、: binary logarithm)とは、2を底とする対数 log2 x のことである。これは、指数関数 x → 2x逆関数でもある。

コンピューターへの応用

[編集]

情報理論

[編集]

二進対数は二進法と密接に関係しているため、計算機科学情報理論でしばしば使われる。この文脈において、二進対数は「lg x」と表記されることがよくある[1]。同じ関数の別の表記としてときどき(特にドイツ語で)使われるものとして「ld x」があり、これはラテン語の Logarithmus Duālis から来ている[2]。ただし、ISO 80000-2では「lg x」は log10 x すなわち常用対数を示すとされており、二進対数の略記法は「lb x」である。本稿でもこれに従う。

正整数 n二進法における桁数(すなわちビット数)は 1 + lb n の整数部分であり、以下の床関数で表される。

計算の複雑性

[編集]

二進対数は、アルゴリズム解析で頻出する。1より大きな数 n を2で繰り返し割っていき、その値を1以下にするために必要な繰り返し回数は、 で与えられる。この発想は、多くのアルゴリズムデータ構造の分析で使用される。例えば二分検索では、検索すべき空間の大きさは操作の繰り返しごとに半分になる。ゆえに、大きさ1の問題を得るには大まかにいって lb n 回の繰り返しが必要となり、そのあとは定数時間で終了する。同様に、n 個の要素からなる完全平衡二分探索木は、1 + lb n の高さをもつ。

しかし、アルゴリズムの実行時間は通常、定数倍の差を無視してランダウの記号で表記される。ここで、1以外の任意の正の数 k に対して log2 n = logk n / logk 2(底の変換)が成り立つので、O(log2 n) の実行時間をもつアルゴリズムは、1より大きな任意の底、たとえば13を用いて、O(log13 n) の実行時間を持つとも表現できる[3]。したがって、O(log n)O(n log n) といった式では、対数の底がいくつであるかは本質的な問題ではない。

ただし、対数の底を指定する必要があるケースもあるので注意が必要である。例えば、所要時間 O(2lb n) の計算と、所要時間 O(2ln n) の計算とは同等ではない。前者は O(n) と等価であり、後者は O(n0.6931...) と等価である。

O(n log n) の実行時間をもつアルゴリズムは、しばしば線形対数と呼ばれる。O(log n)O(n log n) の実行時間をもつアルゴリズムの例としては、次のようなものがある。

電卓の使用法

[編集]

log2 のボタンがない電卓で log2 n を計算するための簡単な方法は、関数電卓に一般的に存在する自然対数 "ln" または常用対数 "Log" のボタンを使うことである。この場合、次のような底の変換公式を使うことになる。

log2 n = ln n / ln 2 = Log n / Log 2

従って、

log2 n = logen × 1.442695... = log10 n × 3.321928...

となる。

ところでこの式は、loge n + log10 n が0.6%以内の差で log2 n と一致する、という興味深い結果を与える。実際のところ、loge n + log10 n という式は、

という底を用いて、log2.0081359... n と表現される。

二進対数の算出

[編集]

整数→整数

[編集]

小数点以下の切り上げ・切り捨てを行って、整数→整数の二進対数を定めることができる。これら二つ(切り上げ・切り下げ)の整数二進対数の間には、

  ただし、1 ≦ n

の関係がある[4]。この左辺の関数は、 とおくことによって、定義域を n ≧ 0 にまで拡張できる。このように拡張した関数は、非負整数 nm ビット符号なし二進表示における先頭の0の個数英語版 nlz(n) との間で

[4]

の関係にある。この整数二進対数は、n の最上位ビットがどこにあるかを示している。

実数→実数

[編集]

一般の正の実数に対しては、二進対数は次の2段階の手順で計算できる。

  1. 整数部分 を計算する。
  2. 小数部分を計算する。

まず、整数部分の計算は簡単である。任意の正の実数 x に対して、2nx < 2n+1 となるような整数 n が唯一に定まる[5]。この各辺を 2n で割った 1 ≦ x/2n < 2 という式を立ててもよい。これをもって、二進対数の整数部分を n と定める。そして、この n を使って、小数部分を lb (x/2n) と表記することにする。すなわち、y = x/2n と置くと、次のようになる。

lb x = n + lb y  ただし、1 ≦ y < 2

小数部分 lb y は、掛け算と割り算のみを使って再帰的に計算できる。この計算手順は以下のとおりとなる。

  1. まず、1 ≦ y < 2 から出発する。y = 1 ならば、小数部分は0となって、その時点で終了である。
  2. y > 1 ならば、y を繰り返し2乗して、2 ≦ z < 4 なる実数 z を得る。2乗した回数を m とすると、 となる。
  3. この式の両辺の対数をとり、式変形を行うと次のようになる。
  4. この m の値を記録しておく。
  5. 2乗する作業をやめる基準は 2 ≦ z < 4 であった。したがって、1 ≦ z/2 < 2 となっている。そこであらためて y := z/2 と置き、この新しい y の二進対数を同じ手法で計算する。

そして最終的に、lb x を次のように計算する。以下、m[i] は、アルゴリズムの i 回目の繰り返しにおいて2乗の操作を行った回数とする。

ある時点で y = 1 となった場合には、この計算は当然、そこまでで終了する。逆に、永久に y = 1 とならない場合には、この式は無限級数となるが、すべての i について m[i] ≧ 1 が成り立つので、どの項もその直前の項より小さくなっている。よって、比較判定法により、この級数が必ず収束するということがわかる。

実用上は、計算に無限の時間を費やすわけにはいかないので、計算を途中で打ち切った近似値を使うことになる。級数の i 番目の項より後ろを切り捨てた場合の誤差の上限は、(1/2)m[1]+m[2]+…+m[i] である。

しかし実際には幸いなことに、このような高コストな計算も、無限級数の切り捨ても必要とせずに、

  1. 値を2乗する。
  2. 結果が2以上であれば、2で割る。

という計算を繰り返すのみで簡単に対数を得ることができる。具体的なコードを Microsoft Visual Basic で記述すると、下記のとおりとなる。(なお、この実装ではわかりやすさのために、関数の戻り値を2進表現の文字列としてある。当然ながら、その後の計算のためには、何らかの方法で数値型のデータにしなければならない。)

Function lb(ByVal y As Double, ByVal numDigits As Integer) as String
    Dim result As String
    result = "0."
    If y < 1 Or 2 <= y Then
        lb = "1≦y<2の値を渡してください。"
        Exit Function
    End If
    While numDigits > 0
        y = y * y
        If 2 <= y Then
            result = result & "1": y = y / 2
        Else
            result = result & "0"
        End If
        numDigits = numDigits - 1
    Wend
    lb = result
End Function

例として、1.65の二進対数を4ビットの精度で計算する(すなわち、上記の関数を lb(1.65, 4) として呼び出す)ケースを考える。このプログラムを逐次追いかけていくと次のようになる。

  • まず、このプログラムでは整数位の計算が既に終わっていることを前提とする(すなわち、1 ≦ y < 2 となっていることを要求する)ので、無条件で "0." の2文字を書く。
  • 与えられた y = 1.65 を2乗すると2.72となる。これは2以上なので、小数1桁目として1を書く。この2.72を2で割って1.36を得る。
  • 1.36を2乗すると1.85となる。これは2より小さいので、2で割ることはせず、2桁目として0を書く。
  • 1.85を再度2乗すると3.43となる。これは2以上なので、3桁目として1を書く。この3.43を2で割って1.72を得る。
  • 1.72を2乗すると2.95となる。これは2以上なので、4桁目として1を書く。ほしい精度は4ビットなので、これで計算終了とする。

こうして0.1011という数字列を得たので、

lb 1.65 ≒ 0.1011(2) = 13/16

と確定させる。このとき、誤差は1/16未満となっている。さらにもう1ビット計算すれば27/32となり、誤差は1/32未満となる。一般に、m ビットの精度がほしい(すなわち、誤差を (1/2)1+m 未満としたい)ときには、2乗の計算をちょうど m 回、2で割る計算を最大 m 回行えば必要十分である。

関連項目

[編集]

脚注

[編集]
  1. ^ Cormen, Thomas H.; Leiserson, Charles E., Rivest, Ronald L., Stein, Clifford (2001) [1990]. Introduction to Algorithms (2nd ed.). MIT Press and McGraw-Hill. p. 34. ISBN 0-262-03293-7 
  2. ^ 例えば、次を参照。Bauer, Friedrich L. (2009), Origins and Foundations of Computing: In Cooperation with Heinz Nixdorf MuseumsForum, Springer Science & Business Media, p. 54, ISBN 9783642029929, https://books.google.co.jp/books?id=y4uTaLiN-wQC&pg=PA54&redir_esc=y&hl=ja .
  3. ^ 1より小さな底でも対数の算出自体は当然ながら可能である。しかし、そのような底を用いると n > 1 のときに log n < 0、特に、n → +∞ のときに log n → −∞ となるため、所要時間の評価用としては実用的でない。
  4. ^ a b Warren Jr., Henry S. (2002). Hacker's Delight. Addison Wesley. pp. 215. ISBN 978-0-201-91465-8 
  5. ^ x < 1 であっても n が定まることに注意。このときの n は負の数である。