二つ値を返せば良いんですよ。メモ化なんてしなくていい。取りあえず Scheme の多値でやってみました。速い。。どうして?
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
;; http://d.hatena.ne.jp/mrkn/20100906/easy_faster_fib | |
(define (fib-i n) | |
(if (or (zero? n)(= n 1)) | |
(values 1 1) | |
(receive (cur prev) | |
(fib-i (- n 1)) | |
(values (+ cur prev) cur)))) | |
(fib-i 10) | |
;; 89 | |
;; 55 | |
(time (fib-i 10000)) | |
;(time (fib-i 10000)) | |
; real 0.094 | |
; user 0.047 | |
; sys 0.047 |
あとはまぁ、取りあえず参考までに。
メモ化。昨日のメモ化マクロを使ってみました。
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
;; memoize | |
;; http://valvallow.blogspot.com/2010/09/paip-memo-memoize-define-memo.html | |
(define-memo (fib-m n) | |
(if (or (zero? n) | |
(= n 1)) | |
1 | |
(+ (fib-m (- n 1)) | |
(fib-m (- n 2))))) | |
(fib-m 10) | |
;; 89 | |
(time (fib-m 10000)) | |
;(time (fib-m 10000)) | |
; real 0.125 | |
; user 0.125 | |
; sys 0.000 |
遅延ストリーム。
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
;; stream | |
;; http://valvallow.blogspot.com/2010/06/fib-tail-call-fib-lazy-fib.html | |
(use util.stream) | |
(define fib-s (stream-cons 1 (stream-cons 1 (stream-map + fib-s (stream-cdr fib-s))))) | |
(stream-ref fib-s 10) | |
;; 89 | |
(time (stream-ref fib-s 10000)) | |
;(time (stream-ref fib-s 10000)) | |
; real 0.172 | |
; user 0.157 | |
; sys 0.016 |
末尾再帰。ようはループですわな。
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
;; tail call | |
;; http://valvallow.blogspot.com/2010/06/fib-tail-call-fib-lazy-fib.html | |
(define (fib-t n) | |
(let rec ((cur 1)(next 2)(n n)) | |
(if (or (zero? n) | |
(= n 1)) | |
cur | |
(rec next (+ cur next)(- n 1))))) | |
(fib-t 10) | |
;; 89 | |
(time (fib-t 10000)) | |
;(time (fib-t 10000)) | |
; real 0.109 | |
; user 0.047 | |
; sys 0.047 |
普通の再帰。これは 10000 なんてとても試す気になれませんね。。
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
;; normal | |
(define (fib n) | |
(if (or (zero? n) | |
(= n 1)) | |
1 | |
(+ (fib (- n 1)) | |
(fib (- n 2))))) | |
(fib 10) | |
;; 89 | |
(time (fib 35)) | |
;(time (fib 35)) | |
; real 2.094 | |
; user 2.078 | |
; sys 0.000 | |
;; (time (fib 10000)) | |
;; ... |
fib-i なぞ・・・。
追記
わかりました!というか教えて頂きました。末尾再帰の例と同じような計算の仕方だからですね。末尾再帰ではないし、末尾再帰のようにその都度計算を行うのではなく再帰を戻りながら最後にまとめて計算する点は違えども、計算量は末尾再帰の例と同じということですよね。
というか、末尾再帰とか多値とかそういう話じゃないわけですね。。なんか、すいません。
ありがとうございました!
追記2
爆速過ぎワロタ
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(define (fib-l n) | |
(let rec ((a 1)(b 1)(p 0)(q 1)(count n)) | |
(cond ((= count 0) b) | |
((even? count) | |
(rec a | |
b | |
(+ (* p p) (* q q)) | |
(+ (* 2 p q) (* q q)) | |
(/ count 2))) | |
(else (rec (+ (* b q) (* a q) (* a p)) | |
(+ (* b p) (* a q)) | |
p | |
q | |
(- count 1)))))) | |
(time (fib-l 10000)) | |
;(time (fib-l 10000)) | |
; real 0.000 | |
; user 0.000 | |
; sys 0.000 | |
(time (fib-l 100000)) | |
;(time (fib-l 100000)) | |
; real 0.203 | |
; user 0.203 | |
; sys 0.000 |
0 件のコメント:
コメントを投稿