Scheme:多値の分配
Scheme:多値で出た議論
セマンティックスとか継続との絡みは良く分かってないのですが、 対称性という意味でいうと、
(func0 (func1 arg1) (func2 arg2) (func3 arg3) ...) => (func0 val1 val2 val3 ...)
という風に複数の式からの返り値を引数にとる。 これが n-in だとすると
(???? func1 func2 func3 ... (values val1 val2 val3 ...))
=> どういう動作になるかは分からないけど UNIX の fork みたいな感じで
(func1 val1) (func2 val2) (func3 val3) ....
というスレッドの分岐みたいのが起こるのが m-out な気がしちゃうなぁ。
(apply func arg1 arg2 ...)
という n-in に対して
(apply-val-to-funcs values func1 func2 func3 ...)
で、 values が多値を返す式、 とすると apply-val-to-funcs は call-with-values に相当するやつみたいなイメージなんだけど、個人的には。 values の返す複数の値が順に func1 func2 ... にばらまかれるような感じ。 cut-sea:2004/06/03 14:32:51 PDT
実装
teranishiさんによる実装例
(define-syntax call-each-values (syntax-rules () ((_ (func funcs ...) args ...) (call-each-values "gensym" () () (args ...) (funcs ...) (func funcs ...) (args ...))) ((_ "gensym" l (lr ...) () () (func ...) (args ...)) (call-each-values "bind" (l lr ...) (l lr ...) (func ...) (args ...))) ((_ "gensym" l (lr ...) () (f g ...) (func ...) (args ...)) (call-each-values "gensym" () (l lr ...) (args ...) (g ...) (func ...) (args ...))) ((_ "gensym" (tr ...) (lr ...) (a b ...) (f ...) (func ...) (args ...)) (call-each-values "gensym" (t tr ...) (lr ...) (b ...) (f ...) (func ...) (args ...))) ((_ "bind" (() ...) ((l ...) ...) (func ...) ()) (values (func l ...) ...)) ((_ "bind" ((v vr ...) ...) ((l ...) ...) (func ...) (arg args ...)) (receive (v ...) arg (call-each-values "bind" ((vr ...) ...) ((l ...) ...) (func ...) (args ...))))))
使用例
(define (qr-list a b) (if (or (null? a) (null? b)) (values () ()) (call-each-values (cons cons) (quotient&remainder (car a) (car b)) (qr-list (cdr a) (cdr b))))) gosh> (qr-list '(10 9 8 7 6) '(5 4 3 2 1)) (2 2 2 3 6) (0 1 2 1 0)
見てくれが通常の再帰に近くなるし、一時変数を使う必要もありません。
- teranishi: let-valuesではなくreceiveを使うように変更 (2004/06/06 06:45:13 PDT)
Tag: 多値