Merge branch 'stibear-lazy'

This commit is contained in:
Yuichi Nishiwaki 2016-06-03 14:35:13 +09:00
commit dfc72fd011
2 changed files with 40 additions and 26 deletions

View File

@ -1,40 +1,45 @@
;;; Appendix A. Standard Libraries Lazy ;;; Appendix A. Standard Libraries Lazy
(define-library (scheme lazy) (define-library (scheme lazy)
(import (scheme base) (import (scheme base))
(picrin macro))
;; type 'a <promise> = cached of 'a | chained of 'a promise | pending of () -> 'a promise
(define-record-type <promise> (define-record-type <promise>
(make-promise% done value) (promise state value)
promise? promise?
(done promise-done? set-promise-done!) (state promise-state set-promise-state!)
(value promise-value set-promise-value!)) (value promise-value set-promise-value!))
(define-syntax delay-force
(syntax-rules ()
((_ expr)
(make-promise% #f (lambda () expr)))))
(define-syntax delay
(syntax-rules ()
((_ expr)
(delay-force (make-promise% #t expr)))))
(define (force promise)
(if (promise-done? promise)
(promise-value promise)
(let ((new-promise ((promise-value promise))))
(if (promise-done? promise)
(promise-value promise)
(begin
(set-promise-done! promise (promise-done? new-promise))
(set-promise-value! promise (promise-value new-promise))
(force promise))))))
(define (make-promise obj) (define (make-promise obj)
(if (promise? obj) (if (promise? obj)
obj obj
(make-promise% #t obj))) (promise 'cached obj)))
(define-syntax delay-force
(syntax-rules ()
((_ expr)
(promise (string->symbol "pending") (lambda () expr)))))
(define-syntax delay
(syntax-rules ()
((_ expr)
(delay-force (make-promise expr)))))
(define (force p)
(let ((v (promise-value p)))
(case (promise-state p)
((cached) v)
((chained) (let ()
(when (eq? 'cached (promise-state v))
(set-promise-state! p 'cached)
(set-promise-value! p (promise-value v)))
(force v)))
((pending) (let ((q (v)))
(when (eq? 'pending (promise-state p))
(set-promise-state! p 'chained)
(set-promise-value! p q))
(force p))))))
(export delay-force (export delay-force
delay delay

View File

@ -363,6 +363,15 @@
(force x) (force x)
(promise? x))) (promise? x)))
(let ()
(define x 0)
(define np (delay (begin
(set! x (+ x 1))
(if (= x 1) 'ok 'ng))))
(define op (delay-force np))
(force op)
(test 'ok (force np)))
(let () (let ()
(define flag #f) (define flag #f)
(define p (delay (if flag (define p (delay (if flag