759 lines
20 KiB
Scheme
759 lines
20 KiB
Scheme
(define-library (srfi 1)
|
|
(import (scheme base)
|
|
(scheme cxr))
|
|
|
|
;; # Constructors
|
|
;; cons list
|
|
;; xcons cons* make-list list-tabulate
|
|
;; list-copy circular-list iota
|
|
(define (xcons a b)
|
|
(cons b a))
|
|
|
|
;; means for inter-referential definition
|
|
(define append-reverse #f)
|
|
|
|
(define (cons* x . args)
|
|
(let rec ((acc '()) (x x) (lst args))
|
|
(if (null? lst)
|
|
(append-reverse acc x)
|
|
(rec (cons x acc) (car lst) (cdr lst)))))
|
|
|
|
(define (list-tabulate n init-proc)
|
|
(let rec ((acc '()) (n (- n 1)))
|
|
(if (zero? n)
|
|
(cons n acc)
|
|
(rec (cons n acc) (- n 1)))))
|
|
|
|
(define (circular-list elt . args)
|
|
(let ((lst (cons elt args)))
|
|
(let rec ((l lst))
|
|
(if (null? (cdr l))
|
|
(set-cdr! l lst)
|
|
(rec (cdr l))))
|
|
lst))
|
|
|
|
(define (iota count . lst)
|
|
(let ((start (if (pair? lst) (car lst) 0))
|
|
(step (if (and (pair? lst) (pair? (cdr lst)))
|
|
(cadr lst) 1)))
|
|
(let rec ((count (- count 1)) (acc '()))
|
|
(if (zero? count)
|
|
(cons start acc)
|
|
(rec (- count 1)
|
|
(cons (+ start (* count step)) acc))))))
|
|
|
|
(export cons list xcons make-list list-tabulate list-copy circular-list iota)
|
|
|
|
;; # Predicates
|
|
;; pair? null?
|
|
;; proper-list? circular-list? dotted-list?
|
|
;; not-pair? null-list?
|
|
;; list=
|
|
(define (not-pair? x)
|
|
(not (pair? x)))
|
|
;; detects circular list using Floyd's cycle-finding algorithm
|
|
(define (circular-list? x)
|
|
(let rec ((rapid x) (local x))
|
|
(if (and (pair? rapid) (pair? (cdr rapid)))
|
|
(if (eq? (cddr rapid) (cdr local))
|
|
#t
|
|
(rec (cddr rapid) (cdr local)))
|
|
#f)))
|
|
|
|
(define proper-list? list?)
|
|
|
|
(define (dotted-list? x)
|
|
(and (pair? x)
|
|
(not (proper-list? x))
|
|
(not (circular-list? x))))
|
|
|
|
(define (null-list? x)
|
|
(cond ((pair? x) #f)
|
|
((null? x) #t)
|
|
(else (error "null-list?: argument out of domain" x))))
|
|
|
|
(define (list= elt= . lists)
|
|
(or (null? lists)
|
|
(let rec1 ((list1 (car lists)) (others (cdr lists)))
|
|
(or (null? others)
|
|
(let ((list2 (car others))
|
|
(others (cdr others)))
|
|
(if (eq? list1 list2)
|
|
(rec1 list2 others)
|
|
(let rec2 ((l1 list1) (l2 list2))
|
|
(if (null-list? l1)
|
|
(and (null-list? l2)
|
|
(rec1 list2 others))
|
|
(and (not (null-list? l2))
|
|
(elt= (car l1) (car l2))
|
|
(rec2 (cdr l1) (cdr l2)))))))))))
|
|
|
|
(export pair? null? not-pair? proper-list? circular-list? null-list? list=)
|
|
|
|
;; # Selectors
|
|
;; car cdr ... cddadr cddddr list-ref
|
|
;; first second third fourth fifth sixth seventh eighth ninth tenth
|
|
;; car+cdr
|
|
;; take drop
|
|
;; take-right drop-right
|
|
;; take! drop-right!
|
|
;; split-at split-at!
|
|
;; last last-pair
|
|
(define (car+cdr pair)
|
|
(values (car pair) (cdr pair)))
|
|
|
|
(define (take x i)
|
|
(if (zero? i)
|
|
'()
|
|
(cons (car x)
|
|
(take (cdr x) (- i 1)))))
|
|
|
|
(define (drop x i)
|
|
(if (zero? i)
|
|
x
|
|
(drop (cdr x) (- i 1))))
|
|
|
|
(define (take-right flist i)
|
|
(let ((len (length flist)))
|
|
(drop flist (- len i))))
|
|
|
|
(define (drop-right flist i)
|
|
(let ((len (length flist)))
|
|
(take flist (- len i))))
|
|
|
|
(define (take! x i)
|
|
(let rec ((lis x) (n (- i 1)))
|
|
(if (zero? n)
|
|
(begin (set-cdr! lis '()) x)
|
|
(rec (cdr lis) (- n 1)))))
|
|
|
|
(define (drop-right! flist i)
|
|
(let ((lead (drop flist i)))
|
|
(if (not-pair? lead)
|
|
'()
|
|
(let rec ((lis1 flist) (lis2 (cdr lead)))
|
|
(if (pair? lis2)
|
|
(rec (cdr lis1) (cdr lis2))
|
|
(begin (set-cdr! lis1 '()) flist))))))
|
|
|
|
(define (split-at x i)
|
|
(values (take x i) (drop x i)))
|
|
|
|
(define (split-at! x i)
|
|
(values (take! x i) (drop x i)))
|
|
|
|
(define (last pair)
|
|
(car (take-right pair 1)))
|
|
|
|
(define (last-pair pair)
|
|
(take-right pair 1))
|
|
|
|
(define first car)
|
|
(define second cadr)
|
|
(define third caddr)
|
|
(define fourth cadddr)
|
|
(define (fifth pair)
|
|
(list-ref pair 4))
|
|
(define (sixth pair)
|
|
(list-ref pair 5))
|
|
(define (seventh pair)
|
|
(list-ref pair 6))
|
|
(define (eighth pair)
|
|
(list-ref pair 7))
|
|
(define (ninth pair)
|
|
(list-ref pair 8))
|
|
(define (tenth pair)
|
|
(list-ref pair 9))
|
|
|
|
|
|
(export car cdr car+cdr list-ref
|
|
caar cadr cdar cddr caaar caadr cadar caddr cdaar cdadr cddar cdddr
|
|
caaaar caaadr caadar caaddr cadaar cadadr caddar cadddr cdaaar cdaadr
|
|
cdadar cdaddr cddaar cddadr cdddar cddddr
|
|
first second third fourth fifth sixth seventh eighth ninth tenth
|
|
take drop take-right drop-right take! drop-right!
|
|
split-at split-at! last last-pair)
|
|
|
|
;; # Miscellaneous
|
|
;; length length+
|
|
;; append concatenate reverse
|
|
;; append! concatenate! reverse!
|
|
;; append-reverse append-reverse!
|
|
;; zip unzip1 unzip2 unzip3 unzip4 unzip5
|
|
;; count
|
|
(define (length+ lst)
|
|
(if (not (circular-list? lst))
|
|
(length lst)))
|
|
|
|
(define (concatenate lists)
|
|
(apply append lists))
|
|
|
|
(define (append! . lists)
|
|
(if (null? lists)
|
|
'()
|
|
(let rec ((lst lists))
|
|
(if (not-pair? (cdr lst))
|
|
(car lst)
|
|
(begin (set-cdr! (last-pair (car lst)) (cdr lst))
|
|
(rec (cdr lst)))))))
|
|
|
|
(define (concatenate! lists)
|
|
(apply append! lists))
|
|
|
|
(define (reverse! list)
|
|
(let rec ((lst list) (acc '()))
|
|
(if (null? lst)
|
|
acc
|
|
(let ((rst (cdr lst)))
|
|
(set-cdr! lst acc)
|
|
(rec rst lst)))))
|
|
|
|
(set! append-reverse
|
|
(lambda (rev-head tail)
|
|
(if (null? rev-head)
|
|
tail
|
|
(append-reverse (cdr rev-head) (cons (car rev-head) tail)))))
|
|
|
|
(define (append-reverse! rev-head tail)
|
|
(let ((rst (cdr rev-head)))
|
|
(if (null? rev-head)
|
|
tail
|
|
(begin (set-cdr! rev-head tail)
|
|
(append-reverse! rst rev-head)))))
|
|
|
|
(define (zip . lists)
|
|
(apply map list lists))
|
|
|
|
(define (unzip1 list)
|
|
(map first list))
|
|
|
|
(define (unzip2 list)
|
|
(values (map first list)
|
|
(map second list)))
|
|
|
|
(define (unzip3 list)
|
|
(values (map first list)
|
|
(map second list)
|
|
(map third list)))
|
|
|
|
(define (unzip4 list)
|
|
(values (map first list)
|
|
(map second list)
|
|
(map third list)
|
|
(map fourth list)))
|
|
|
|
(define (unzip5 list)
|
|
(values (map first list)
|
|
(map second list)
|
|
(map third list)
|
|
(map fourth list)
|
|
(map fifth list)))
|
|
|
|
(define (count pred . clists)
|
|
(let rec ((tflst (apply map pred clists)) (n 0))
|
|
(if (null? tflst)
|
|
n
|
|
(rec (cdr tflst) (if (car tflst) (+ n 1) n)))))
|
|
|
|
(export length length+
|
|
append append! concatenate concatenate!
|
|
reverse reverse! append-reverse append-reverse!
|
|
zip unzip1 unzip2 unzip3 unzip4 unzip5
|
|
count)
|
|
|
|
;; # Fold, unfold & map
|
|
;; map for-each
|
|
;; fold unfold pair-fold reduce
|
|
;; fold-right unfold-right pair-fold right reduce-right
|
|
;; append-map append-map!
|
|
;; map! pair-for-each filter-map map-in-order
|
|
|
|
;; means for inter-referential definition
|
|
(define every #f)
|
|
|
|
(define (fold kons knil clist . clists)
|
|
(if (null? clists)
|
|
(let rec ((acc knil) (clist clist))
|
|
(if (null? clist)
|
|
acc
|
|
(rec (kons (car clist) acc) (cdr clist))))
|
|
(let rec ((acc knil) (clists (cons clist clists)))
|
|
(if (every pair? clists)
|
|
(rec (apply kons (append (map car clists) (list acc)))
|
|
(map cdr clists))
|
|
acc))))
|
|
|
|
(define (fold-right kons knil clist . clists)
|
|
(if (null? clists)
|
|
(let rec ((clist clist) (cont values))
|
|
(if (null? clist)
|
|
(cont knil)
|
|
(rec (cdr clist) (lambda (x) (cont (kons (car clist) x))))))
|
|
(let rec ((clists (cons clist clists)) (cont values))
|
|
(if (every pair? clists)
|
|
(rec (map cdr clists)
|
|
(lambda (x)
|
|
(cont (apply kons (append (map car clists) (list x))))))
|
|
(cont knil)))))
|
|
|
|
(define (pair-fold kons knil clist . clists)
|
|
(if (null? clists)
|
|
(let rec ((acc knil) (clist clist))
|
|
(if (null? clist)
|
|
acc
|
|
(let ((tail (cdr clist)))
|
|
(rec (kons clist acc) tail))))
|
|
(let rec ((acc knil) (clists (cons clist clists)))
|
|
(if (every pair? clists)
|
|
(let ((tail (map cdr clists)))
|
|
(rec (apply kons (append clists (list acc)))
|
|
tail))
|
|
acc))))
|
|
|
|
(define (pair-fold-right kons knil clist . clists)
|
|
(if (null? clists)
|
|
(let rec ((clist clist) (cont values))
|
|
(if (null? clist)
|
|
(cont knil)
|
|
(let ((tail (map cdr clists)))
|
|
(rec tail (lambda (x) (cont (kons clist x)))))))
|
|
(let rec ((clists (cons clist clists)) (cont values))
|
|
(if (every pair? clists)
|
|
(let ((tail (map cdr clists)))
|
|
(rec tail
|
|
(lambda (x)
|
|
(cont (apply kons (append clists (list x)))))))
|
|
(cont knil)))))
|
|
|
|
(define (reduce f ridentity list)
|
|
(if (null? list)
|
|
ridentity
|
|
(fold f (car list) (cdr list))))
|
|
|
|
(define (reduce-right f ridentity list)
|
|
(fold-right f ridentity list))
|
|
|
|
(define (unfold p f g seed . tail-gen)
|
|
(let ((tail-gen (if (null? tail-gen)
|
|
(lambda (x) '())
|
|
(car tail-gen))))
|
|
(let rec ((seed seed) (cont values))
|
|
(if (p seed)
|
|
(cont (tail-gen seed))
|
|
(rec (g seed) (lambda (x) (cont (cons (f seed) x))))))))
|
|
|
|
(define (unfold-right p f g seed . tail)
|
|
(let rec ((seed seed) (lst tail))
|
|
(if (p seed)
|
|
lst
|
|
(rec (g seed) (cons (f seed) lst)))))
|
|
|
|
(define (append-map f . clists)
|
|
(apply append (apply map f clists)))
|
|
|
|
(define (append-map! f . clists)
|
|
(apply append! (apply map f clists)))
|
|
|
|
(define (pair-for-each f clist . clists)
|
|
(if (null? clist)
|
|
(let rec ((clist clist))
|
|
(if (pair? clist)
|
|
(begin (f clist) (rec (cdr clist)))))
|
|
(let rec ((clists (cons clist clists)))
|
|
(if (every pair? clists)
|
|
(begin (apply f clists) (rec (map cdr clists)))))))
|
|
|
|
(define (map! f list . lists)
|
|
(if (null? lists)
|
|
(pair-for-each (lambda (x) (set-car! x (f (car x)))) list)
|
|
(let rec ((list list) (lists lists))
|
|
(if (pair? list)
|
|
(let ((head (map car lists))
|
|
(rest (map cdr lists)))
|
|
(set-car! list (apply f (car list) head))
|
|
(rec (cdr list) rest)))))
|
|
list)
|
|
|
|
(define (map-in-order f clist . clists)
|
|
(if (null? clists)
|
|
(let rec ((clist clist) (acc '()))
|
|
(if (null? clist)
|
|
(reverse! acc)
|
|
(rec (cdr clist) (cons (f (car clist)) acc))))
|
|
(let rec ((clists (cons clist clists)) (acc '()))
|
|
(if (every pair? clists)
|
|
(rec (map cdr clists)
|
|
(cons* (apply f (map car clists)) acc))
|
|
(reverse! acc)))))
|
|
|
|
(define (filter-map f clist . clists)
|
|
(let recur ((l (apply map f clist clists)))
|
|
(cond ((null? l) '())
|
|
((car l) (cons (car l) (recur (cdr l))))
|
|
(else (recur (cdr l))))))
|
|
|
|
(export map for-each
|
|
fold unfold pair-fold reduce
|
|
fold-right unfold-right pair-fold-right reduce-right
|
|
append-map append-map!
|
|
map! pair-for-each filter-map map-in-order)
|
|
|
|
;; # Filtering & partitioning
|
|
;; filter partition remove
|
|
;; filter! partition! remove!
|
|
(define (filter pred list)
|
|
(let ((pcons (lambda (v acc) (if (pred v) (cons v acc) acc))))
|
|
(reverse (fold pcons '() list))))
|
|
|
|
(define (remove pred list)
|
|
(filter (lambda (x) (not (pred x))) list))
|
|
|
|
(define (partition pred list)
|
|
(values (filter pred list)
|
|
(remove pred list)))
|
|
|
|
(define (filter! pred list)
|
|
(let rec ((lst list))
|
|
(if (null? lst)
|
|
lst
|
|
(if (pred (car lst))
|
|
(begin (set-cdr! lst (rec (cdr lst)))
|
|
lst)
|
|
(rec (cdr lst))))))
|
|
|
|
(define (remove! pred list)
|
|
(filter! (lambda (x) (not (pred x))) list))
|
|
|
|
(define (partition! pred list)
|
|
(values (filter! pred list)
|
|
(remove! pred list)))
|
|
|
|
(export filter partition remove
|
|
filter! partition! remove!)
|
|
|
|
;; # Searching
|
|
;; member memq memv
|
|
;; find find-tail
|
|
;; any every
|
|
;; list-index
|
|
;; take-while drop-while take-while!
|
|
;; span break span! break!
|
|
|
|
(define (find-tail pred list)
|
|
(if (null? list)
|
|
#f
|
|
(if (pred (car list))
|
|
list
|
|
(find-tail pred (cdr list)))))
|
|
|
|
(define (find pred list)
|
|
(let ((tail (find-tail pred list)))
|
|
(if tail
|
|
(car tail)
|
|
#f)))
|
|
|
|
(define (take-while pred clist)
|
|
(let rec ((clist clist) (cont values))
|
|
(if (null? clist)
|
|
(cont '())
|
|
(if (pred (car clist))
|
|
(rec (cdr clist)
|
|
(lambda (x) (cont (cons (car clist) x))))
|
|
(cont '())))))
|
|
|
|
(define (take-while! pred clist)
|
|
(let rec ((clist clist))
|
|
(if (null? clist)
|
|
'()
|
|
(if (pred (car clist))
|
|
(begin (set-cdr! clist (rec (cdr clist)))
|
|
clist)
|
|
'()))))
|
|
|
|
(define (drop-while pred clist)
|
|
(let rec ((clist clist))
|
|
(if (null? clist)
|
|
'()
|
|
(if (pred (car clist))
|
|
(rec (cdr clist))
|
|
clist))))
|
|
|
|
(define (span pred clist)
|
|
(values (take-while pred clist)
|
|
(drop-while pred clist)))
|
|
|
|
(define (span! pred clist)
|
|
(values (take-while! pred clist)
|
|
(drop-while pred clist)))
|
|
|
|
(define (break pred clist)
|
|
(values (take-while (lambda (x) (not (pred x))) clist)
|
|
(drop-while (lambda (x) (not (pred x))) clist)))
|
|
|
|
(define (break! pred clist)
|
|
(values (take-while! (lambda (x) (not (pred x))) clist)
|
|
(drop-while (lambda (x) (not (pred x))) clist)))
|
|
|
|
(define (any pred clist . clists)
|
|
(if (null? clists)
|
|
(let rec ((clist clist))
|
|
(if (pair? clist)
|
|
(or (pred (car clist))
|
|
(rec (cdr clist)))))
|
|
(let rec ((clists (cons clist clists)))
|
|
(if (every pair? clists)
|
|
(or (apply pred (map car clists))
|
|
(rec (map cdr clists)))))))
|
|
|
|
(set! every
|
|
(lambda (pred clist . clists)
|
|
(if (null? clists)
|
|
(let rec ((clist clist))
|
|
(or (null? clist)
|
|
(if (pred (car clist))
|
|
(rec (cdr clist)))))
|
|
(let rec ((clists (cons clist clists)))
|
|
(or (any null? clists)
|
|
(if (apply pred (map car clists))
|
|
(rec (map cdr clists))))))))
|
|
|
|
(define (list-index pred clist . clists)
|
|
(if (null? clists)
|
|
(let rec ((clist clist) (n 0))
|
|
(if (pair? clist)
|
|
(if (pred (car clist))
|
|
n
|
|
(rec (cdr clist) (+ n 1)))))
|
|
(let rec ((clists (cons clist clists)) (n 0))
|
|
(if (every pair? clists)
|
|
(if (apply pred (map car clists))
|
|
n
|
|
(rec (map cdr clists) (+ n 1)))))))
|
|
|
|
(export member memq memv
|
|
find find-tail
|
|
any every
|
|
list-index
|
|
take-while drop-while take-while!
|
|
span break span! break!)
|
|
|
|
;; # Deleting
|
|
;; delete delete-duplicates
|
|
;; delete! delete-duplicates!
|
|
(define (delete x list . =)
|
|
(let ((= (if (null? =) equal? (car =))))
|
|
(remove (lambda (a) (= x a)) list)))
|
|
|
|
(define (delete! x list . =)
|
|
(let ((= (if (null? =) equal? (car =))))
|
|
(remove! (lambda (a) (= x a)) list)))
|
|
|
|
(define (delete-duplicates list . =)
|
|
(let ((= (if (null? =) equal? (car =))))
|
|
(let rec ((list list))
|
|
(if (null? list)
|
|
list
|
|
(let* ((x (car list))
|
|
(rest (cdr list))
|
|
(deleted (rec (delete x list =))))
|
|
(if (eq? rest deleted) list (cons x deleted)))))))
|
|
|
|
(define (delete-duplicates! list . =)
|
|
(let ((= (if (null? =) equal? (car =))))
|
|
(let rec ((list list))
|
|
(if (null? list)
|
|
list
|
|
(let* ((x (car list))
|
|
(rest (cdr list))
|
|
(deleted (rec (delete! x list =))))
|
|
(if (eq? rest deleted) list (cons x deleted)))))))
|
|
|
|
(export delete delete-duplicates
|
|
delete! delete-duplicates!)
|
|
|
|
;; # Association lists
|
|
;; assoc assq assv
|
|
;; alist-cons alist-copy
|
|
;; alist-delete alist-delete!
|
|
(define (alist-cons key datum alist)
|
|
(cons (cons key datum) alist))
|
|
|
|
(define (alist-copy alist)
|
|
(map (lambda (elt) (cons (car elt) (cdr elt))) alist))
|
|
|
|
(define (alist-delete key alist . =)
|
|
(let ((= (if (null? =) equal? (car =))))
|
|
(remove (lambda (x) (= key (car x))) alist)))
|
|
|
|
(define (alist-delete! key alist . =)
|
|
(let ((= (if (null? =) equal? (car =))))
|
|
(remove! (lambda (x) (= key (car x))) alist)))
|
|
|
|
(export assoc assq assv
|
|
alist-cons alist-copy
|
|
alist-delete alist-delete!)
|
|
|
|
;; # Set operations on lists
|
|
;; lset<= lset= lset-adjoin
|
|
;; lset-union lset-union!
|
|
;; lset-intersection lset-intersection!
|
|
;; lset-difference lset-difference!
|
|
;; lset-xor lset-xor!
|
|
;; lset-diff+intersenction lset-diff+intersection!
|
|
(define (lset<= = . lists)
|
|
(or (null? lists)
|
|
(let rec ((head (car lists)) (rest (cdr lists)))
|
|
(or (null? rest)
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(and (or (eq? head next)
|
|
(every (lambda (x) (member x next =)) head))
|
|
(rec next rest)))))))
|
|
|
|
(define (lset= = . lists)
|
|
(or (null? lists)
|
|
(let rec ((head (car lists)) (rest (cdr lists)))
|
|
(or (null? rest)
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(and (or (eq? head next)
|
|
(and (every (lambda (x) (member x next =)) head)
|
|
(every (lambda (x) (member x head =)) next))
|
|
(rec next rest))))))))
|
|
|
|
(define (lset-adjoin = list . elts)
|
|
(let rec ((list list) (elts elts))
|
|
(if (null? elts)
|
|
list
|
|
(if (member (car elts) list)
|
|
(rec list (cdr elts))
|
|
(rec (cons (car elts) list) (cdr elts))))))
|
|
|
|
(define (lset-union = . lists)
|
|
(if (null? lists)
|
|
lists
|
|
(let rec ((head (car lists)) (rest (cdr lists)))
|
|
(if (null? rest)
|
|
head
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(if (eq? head next)
|
|
(rec head rest)
|
|
(rec (apply lset-adjoin = head next) rest)))))))
|
|
|
|
(define (lset-intersection = . lists)
|
|
(if (null? lists)
|
|
lists
|
|
(let rec ((head (car lists)) (rest (cdr lists)))
|
|
(if (null? rest)
|
|
head
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(if (eq? head next)
|
|
(rec head rest)
|
|
(rec (filter (lambda (x) (member x next =)) head)
|
|
rest)))))))
|
|
|
|
(define (lset-difference = list . lists)
|
|
(let rec ((head list) (rest lists))
|
|
(if (null? rest)
|
|
head
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(if (eq? head next)
|
|
'()
|
|
(rec (remove (lambda (x) (member x next =)) head)
|
|
rest))))))
|
|
|
|
(define (lset-xor = . lists)
|
|
(if (null? lists)
|
|
lists
|
|
(let rec ((head (car lists)) (rest (cdr lists)))
|
|
(if (null? rest)
|
|
head
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(if (eq? head next)
|
|
'()
|
|
(rec (append (remove (lambda (x) (member x next =)) head)
|
|
(remove (lambda (x) (member x head =)) next))
|
|
rest)))))))
|
|
|
|
(define (lset-diff+intersection = list . lists)
|
|
(values (apply lset-difference = list lists)
|
|
(lset-intersection = list (apply lset-union lists))))
|
|
|
|
(define (lset-adjoin! = list . elts)
|
|
(let rec ((list list) (elts elts))
|
|
(if (null? elts)
|
|
list
|
|
(if (member (car elts) list)
|
|
(rec list (cdr elts))
|
|
(let ((tail (cdr elts)))
|
|
(set-cdr! elts list)
|
|
(rec elts tail))))))
|
|
|
|
(define (lset-union! = . lists)
|
|
(letrec ((adjoin
|
|
(lambda (lst1 lst2)
|
|
(if (null? lst2)
|
|
lst1
|
|
(if (member (car lst2) lst1 =)
|
|
(adjoin lst1 (cdr lst2))
|
|
(let ((tail (cdr lst2)))
|
|
(set-cdr! lst2 lst1)
|
|
(adjoin lst2 tail)))))))
|
|
(if (null? lists)
|
|
lists
|
|
(let rec ((head (car lists)) (rest (cdr lists)))
|
|
(if (null? rest)
|
|
head
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(if (eq? head next)
|
|
(rec head rest)
|
|
(rec (adjoin head next) rest))))))))
|
|
|
|
(define (lset-intersection! = . lists)
|
|
(if (null? lists)
|
|
lists
|
|
(let rec ((head (car lists)) (rest (cdr lists)))
|
|
(if (null? rest)
|
|
head
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(if (eq? head next)
|
|
(rec head rest)
|
|
(rec (filter! (lambda (x) (member x next =)) head)
|
|
rest)))))))
|
|
|
|
(define (lset-difference! = list . lists)
|
|
(let rec ((head list) (rest lists))
|
|
(if (null? rest)
|
|
head
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(if (eq? head next)
|
|
'()
|
|
(rec (remove! (lambda (x) (member x next =)) head)
|
|
rest))))))
|
|
|
|
(define (lset-xor! = . lists)
|
|
(if (null? lists)
|
|
lists
|
|
(let rec ((head (car lists)) (rest (cdr lists)))
|
|
(if (null? rest)
|
|
head
|
|
(let ((next (car rest)) (rest (cdr rest)))
|
|
(if (eq? head next)
|
|
'()
|
|
(rec (append! (remove! (lambda (x) (member x next =)) head)
|
|
(remove! (lambda (x) (member x head =)) next))
|
|
rest)))))))
|
|
|
|
(define (lset-diff+intersection! = list . lists)
|
|
(values (apply lset-difference! = list lists)
|
|
(lset-intersection! = list (apply lset-union! lists))))
|
|
|
|
(export lset<= lset= lset-adjoin
|
|
lset-union lset-union!
|
|
lset-intersection lset-intersection!
|
|
lset-difference lset-difference!
|
|
lset-xor lset-xor!
|
|
lset-diff+intersection lset-diff+intersection!)
|
|
|
|
;; # Primitive side-effects
|
|
;; set-car! set-cdr!
|
|
(export set-car! set-cdr!))
|