Ho trovato questo codice in Clojure al setaccio fuori primi N numeri primi:Perché Clojure è molto più veloce di mit-scheme per funzioni equivalenti?
(defn sieve [n]
(let [n (int n)]
"Returns a list of all primes from 2 to n"
(let [root (int (Math/round (Math/floor (Math/sqrt n))))]
(loop [i (int 3)
a (int-array n)
result (list 2)]
(if (>= i n)
(reverse result)
(recur (+ i (int 2))
(if (< i root)
(loop [arr a
inc (+ i i)
j (* i i)]
(if (>= j n)
arr
(recur (do (aset arr j (int 1)) arr)
inc
(+ j inc))))
a)
(if (zero? (aget a i))
(conj result i)
result)))))))
Poi ho scritto l'equivalente (credo) il codice nello Schema (io uso mit-scheme)
(define (sieve n)
(let ((root (round (sqrt n)))
(a (make-vector n)))
(define (cross-out t to dt)
(cond ((> t to) 0)
(else
(vector-set! a t #t)
(cross-out (+ t dt) to dt)
)))
(define (iter i result)
(cond ((>= i n) (reverse result))
(else
(if (< i root)
(cross-out (* i i) (- n 1) (+ i i)))
(iter (+ i 2) (if (vector-ref a i)
result
(cons i result))))))
(iter 3 (list 2))))
Il risultati temporizzazione sono: Per Clojure:
(time (reduce + 0 (sieve 5000000)))
"Elapsed time: 168.01169 msecs"
Per mit-schema:
(time (fold + 0 (sieve 5000000)))
"Elapsed time: 3990 msecs"
Qualcuno può dirmi perché mit-scheme è più di 20 volte più lento?
Grazie per fare questo tipo di paragone. Credo che aiuti entrambe le lingue. – octopusgrabbus
Penso che il tuo codice Scheme non sia corretto. In particolare, 'make-vector' riempirà il vettore con' 0', che viene trattato come * true * in Scheme. Cambiarlo in '(make-vector n #f)' produce risultati molto più sensibili. –
(make-vector n) è riempito con #f in mit-scheme. –