すばやく-数値のO(log N)算術演算の場合-N番目のフィボナッチ数を見つける私はそれについて考え、O(N)の間に機能するソリューションのみが頭に浮かぶことに気付きました。 しかし、解決策は後で見つかりました。
便宜上、表記から移動します
data:image/s3,"s3://crabby-images/d1564/d15645711b109571319bddf939f5f7221c7547cb" alt="N"
data:image/s3,"s3://crabby-images/d8660/d8660e123ffc8218d1d511e97358c226428017b4" alt="n"
data:image/s3,"s3://crabby-images/4745d/4745de3a2fe169df9be157a2901becd5feb92b76" alt="\ mathbb {N} _0 = \左\ {0、1、2、3、... \右\}"
data:image/s3,"s3://crabby-images/da01d/da01d1626fc5c1828c75539702429096354bf4b9" alt="\ mathbb {N} _1 = \左\ {1、2、3、... \右\}"
だから解決策
ナッツ[ 1、p。 112 ]は、次の形式の行列IDを提供します。data:image/s3,"s3://crabby-images/944ab/944ab101e56571d2c63acbdb8b30fee80a1b10d5" alt="\ begin {pmatrix} F_ {n + 1}&F_n \\ F_n&F_ {n-1} \ end {pmatrix} = \ begin {pmatrix} 1&1 \\ 1&0 \ end {pmatrix} ^ n"
身元は証明なしで与えられますが、帰納法によって非常に簡単に証明されます。
右側のマトリックスは、Qマトリックスと呼ばれることもあります。
注:
data:image/s3,"s3://crabby-images/181a2/181a2048fe7b43109a5454cb16aba3ff740e480a" alt="Q = \ begin {pmatrix} 1&1 \\ 1&0 \ end {pmatrix}"
アイデンティティからそれを得る
data:image/s3,"s3://crabby-images/dcd18/dcd189073d71eb8c1f030c32b436f58a6eec88cd" alt="F_n = Q ^ {n-1} \左(1,1 \右)"
data:image/s3,"s3://crabby-images/e5839/e5839784d59a6c74cf820170b5a8a08da13581b3" alt="F_n"
data:image/s3,"s3://crabby-images/108fe/108febae3b2a9918d1ef1f807c76065259f750f1" alt="Q ^ {n-1}"
計算以来
data:image/s3,"s3://crabby-images/e5839/e5839784d59a6c74cf820170b5a8a08da13581b3" alt="F_n"
マトリックスを用意しましょう
data:image/s3,"s3://crabby-images/204f2/204f2d22e7326ef036105475fb6301cc9cc61e07" alt="M"
data:image/s3,"s3://crabby-images/3e004/3e004b147aedfad942bc0e6084fb84ec03f2825b" alt="n \ in \ mathbb {N} _1"
data:image/s3,"s3://crabby-images/d8660/d8660e123ffc8218d1d511e97358c226428017b4" alt="n"
data:image/s3,"s3://crabby-images/4d837/4d837e65d3bc6407700ffa8407ec821624de63b7" alt="n = 2 ^ i、i \ in \ mathbb {N} _0"
data:image/s3,"s3://crabby-images/b9335/b9335f2e5188b0a4e949fe3be9e86d91b176caa8" alt="M ^ n"
data:image/s3,"s3://crabby-images/f7fc4/f7fc4b51a23ab7d5e80404646f77d575666e42a6" alt=""
これは以下を参照します:
data:image/s3,"s3://crabby-images/07592/07592d202848b35e84fb296dbd3c4e70e5ebdf13" alt="M ^ n = M ^ {n / 2} \ cdot M ^ {n / 2} = ... = \ prod ^ {n} _ {1} M ^ 1"
したがって、行列を計算するには
data:image/s3,"s3://crabby-images/b9335/b9335f2e5188b0a4e949fe3be9e86d91b176caa8" alt="M ^ n"
data:image/s3,"s3://crabby-images/5d25b/5d25b9c6ecf958acc9cfea1bcae77492349b5c6d" alt="M ^ {n / 2}"
data:image/s3,"s3://crabby-images/5d25b/5d25b9c6ecf958acc9cfea1bcae77492349b5c6d" alt="M ^ {n / 2}"
data:image/s3,"s3://crabby-images/d2d41/d2d416a1a1966d395416b14a67d6b0b4fc9791b4" alt="M ^ {n / 4}"
明らかに、木の高さは
data:image/s3,"s3://crabby-images/50418/50418840995d345eaeb95fed0bf3c05a1b8f7c75" alt="\ log n"
計算時間を見積もる
data:image/s3,"s3://crabby-images/b9335/b9335f2e5188b0a4e949fe3be9e86d91b176caa8" alt="M ^ n"
data:image/s3,"s3://crabby-images/204f2/204f2d22e7326ef036105475fb6301cc9cc61e07" alt="M"
data:image/s3,"s3://crabby-images/a750c/a750c319f2e09d1cd65c4e32720a063e364f4765" alt="O \左(1 \右)"
data:image/s3,"s3://crabby-images/75806/7580694f53f90908e7ee04bfa0da84b952416694" alt="\ log_2n"
data:image/s3,"s3://crabby-images/b9335/b9335f2e5188b0a4e949fe3be9e86d91b176caa8" alt="M ^ n"
data:image/s3,"s3://crabby-images/6098d/6098d50653e0bdfa4d535e1e2b68aa6ab0c452c1" alt="O \左(\ log n \右)"
また、nが2のべき乗でない場合はどうなりますか?
ここで疑問が生じます:もしもdata:image/s3,"s3://crabby-images/d8660/d8660e123ffc8218d1d511e97358c226428017b4" alt="n"
data:image/s3,"s3://crabby-images/d8660/d8660e123ffc8218d1d511e97358c226428017b4" alt="n"
data:image/s3,"s3://crabby-images/e1686/e168606bb9bb011cb259f3c3fc718e35c56bf70d" alt="n = \ sum_ {p \ in P} 2 ^ p"
どこで
data:image/s3,"s3://crabby-images/571cc/571cc99fcc4e5e49c5933069057717949714f7c2" alt="P_n \サブセット\ mathbb {N} _0"
data:image/s3,"s3://crabby-images/d8660/d8660e123ffc8218d1d511e97358c226428017b4" alt="n"
data:image/s3,"s3://crabby-images/d8660/d8660e123ffc8218d1d511e97358c226428017b4" alt="n"
data:image/s3,"s3://crabby-images/8986f/8986fc1501c60d0c3d1e9a86acdcadf8223f4e59" alt="M ^ n = \ prod_ {p \ in P_n} M ^ {2 ^ p}"
一般に、行列積は可換ではありませんが、 乗算中のオペランドの順序は重要ですが、いわゆる 順列行列の可換性が尊重されます。 マトリックス
data:image/s3,"s3://crabby-images/deaf7/deaf7e91f704a76e516e38127b82d58daaa59993" alt="M ^ i"
data:image/s3,"s3://crabby-images/d3d19/d3d190ccfa4f03b0d7248372a0de62ca2d1f841e" alt="M ^ j"
data:image/s3,"s3://crabby-images/68370/68370541da7ad552b39d634421c37e1c4db64f6d" alt="i、j \ in \ mathbb {N} _0"
したがって、計算アルゴリズム
data:image/s3,"s3://crabby-images/b9335/b9335f2e5188b0a4e949fe3be9e86d91b176caa8" alt="M ^ n"
- 分解する
セットの形式での2の累乗の合計
。
- セットのすべての要素を計算する
。
- 計算する
。
このアルゴリズムの実行時間を推定します。
最初のステップは時間内に完了します
data:image/s3,"s3://crabby-images/b5409/b5409a0554fe53acb5a7080d085fea6f80cd9f65" alt="O \左(\左\ lfloor \ log_2n \ rfloor + 1 \右)= O \左(\ log n \右)"
data:image/s3,"s3://crabby-images/f7dc5/f7dc55566ce039d2d865e5928eebbd3463c3a240" alt="\ lfloor log_2 n \ rfloor + 1"
data:image/s3,"s3://crabby-images/d8660/d8660e123ffc8218d1d511e97358c226428017b4" alt="n"
2番目のステップは
data:image/s3,"s3://crabby-images/558cd/558cd3c4cae2312db3eaa668a6e6dbd8cd268f96" alt="O \ left(\ left \ left(\ lfloor \ log_2 n \ rfloor + 1 \ right)\ cdot \ log_2 n \ right)= O \ left(\ left \ log ^ 2 n \ right)"
data:image/s3,"s3://crabby-images/f7dc5/f7dc55566ce039d2d865e5928eebbd3463c3a240" alt="\ lfloor log_2 n \ rfloor + 1"
3番目のステップは
data:image/s3,"s3://crabby-images/df327/df3275c8629327f834cd26f020530947bb56df1a" alt="O \左(\左\ \ lfloor \ log_2 n \ rfloor \右)= O \左(\左\ログn \右)"
data:image/s3,"s3://crabby-images/afdcd/afdcddf4a3200c18465710812be888ddc8842311" alt="\ lfloor log_2 n \ rfloor"
最適化
このアルゴリズムの実行時間を改善することは可能ですか? はい、できます。 2番目のステップでは、セット内の行列を計算する方法data:image/s3,"s3://crabby-images/e1ef3/e1ef376770639ee78d5d0ec208ad4931728b040c" alt="S"
data:image/s3,"s3://crabby-images/f5309/f5309c1178b6174c248352e51c70bce6d74c567a" alt="p"
data:image/s3,"s3://crabby-images/50bad/50bad20958b76fcdbaafbb371e18e3ab98576b22" alt="O \左(\ log n \右)"
data:image/s3,"s3://crabby-images/50bad/50bad20958b76fcdbaafbb371e18e3ab98576b22" alt="O \左(\ log n \右)"
コード
コーディングに移りましょう。 次の2つの理由から、Pythonでアルゴリズムを実装しました。- 擬似コードを置き換えるのに十分な表現力があります。
- 透過的な長演算があります。
これは次のとおりです。
class MatrixFibonacci:
Q = [[1, 1],
[1, 0]]
def __init__(self):
self.__memo = {}
def __multiply_matrices(self, M1, M2):
"""
( 2x2)."""
a11 = M1[0][0]*M2[0][0] + M1[0][1]*M2[1][0]
a12 = M1[0][0]*M2[0][1] + M1[0][1]*M2[1][1]
a21 = M1[1][0]*M2[0][0] + M1[1][1]*M2[1][0]
a22 = M1[1][0]*M2[0][1] + M1[1][1]*M2[1][1]
r = [[a11, a12], [a21, a22]]
return r
def __get_matrix_power(self, M, p):
""" ( p )."""
if p == 1:
return M
if p in self.__memo:
return self.__memo[p]
K = self.__get_matrix_power(M, int(p/2))
R = self.__multiply_matrices(K, K)
self.__memo[p] = R
return R
def get_number(self, n):
""" n-
( n )."""
if n == 0:
return 0
if n == 1:
return 1
# , ,
# .. 62 = 2^5 + 2^4 + 2^3 + 2^2 + 2^0 = 32 + 16 + 8 + 4 + 1.
powers = [int(pow(2, b))
for (b, d) in enumerate(reversed(bin(n-1)[2:])) if d == '1']
# , pythonic: http://pastebin.com/h8cKDkHX
matrices = [self.__get_matrix_power(MatrixFibonacci.Q, p)
for p in powers]
while len(matrices) > 1:
M1 = matrices.pop()
M2 = matrices.pop()
R = self.__multiply_matrices(M1, M2)
matrices.append(R)
return matrices[0][0][0]
mfib = MatrixFibonacci()
for n in range(0, 128):
num = mfib.get_number(n)
print(num)
,
data:image/s3,"s3://crabby-images/44a6d/44a6d4141f7f6f5a4887726b74928f40a209fe7c" alt="F_1,F_2,...,F_n"
def get_number(self, n):
if n == 0:
return 0
a = 0
b = 1
c = 1
for i in range(n-1):
c = a + b
a = b
b = c
return c
.
data:image/s3,"s3://crabby-images/e0f3c/e0f3cc66f20a625471c3329992947b7358c03c3b" alt="a"
data:image/s3,"s3://crabby-images/497cc/497cc93dd45222883a0db731c99940d3f8378699" alt="a=10,20,...,1000"
data:image/s3,"s3://crabby-images/e0f3c/e0f3cc66f20a625471c3329992947b7358c03c3b" alt="a"
MatrixFibonacci
IterationFibonacci
(, ). 10 000
data:image/s3,"s3://crabby-images/a0481/a0481812e86bad6030ded5c0f04032606211a82d" alt="n \in \left [ a-5,a+5 \right )"
data:image/s3,"s3://crabby-images/e5839/e5839784d59a6c74cf820170b5a8a08da13581b3" alt="F_n"
n <tab> T1 <tab> T2
. .
data:image/s3,"s3://crabby-images/16037/16037c51a82ac35fa902ced7e4aeb26eaf5213e6" alt=""
, -
data:image/s3,"s3://crabby-images/a9c2a/a9c2ad9f80c85a0d8e34379d96a249715d97c5b8" alt="n=300"
data:image/s3,"s3://crabby-images/76cab/76cab6cff0416c1a3adbf3841635b38f354e93c0" alt="F_{94}"
. gnuplot — .
P.S. , TeX- . , .