J'ai trouvé ce code dans Clojure pour trier en premier lieu n des nombres premiers :
(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)))))))
J'ai ensuite écrit le code équivalent (je pense) en Scheme (j'utilise 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))))
Les résultats du chronométrage sont : Pour Clojure :
(time (reduce + 0 (sieve 5000000)))
"Elapsed time: 168.01169 msecs"
Pour mit-scheme :
(time (fold + 0 (sieve 5000000)))
"Elapsed time: 3990 msecs"
Quelqu'un peut-il me dire pourquoi mit-scheme est plus de 20 fois plus lent ?
mise à jour : "la différence était en mode iterprété/compilé. Après avoir compilé le code de mit-scheme, il s'exécutait à une vitesse comparable. - abo-abo 30 avr. 12 à 15:43 "