list.scm 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. ; Copyright (c) 1993-2008 by Richard Kelsey and Jonathan Rees. See file COPYING.
  2. ; The (rnrs lists (6)) library.
  3. (define (assert-procedure who obj)
  4. (if (not (procedure? obj))
  5. (assertion-violation who "not a procedure" obj)))
  6. (define (find proc list)
  7. (assert-procedure 'find proc)
  8. (let loop ((list list))
  9. (cond
  10. ((null? list) #f)
  11. ((proc (car list)) (car list))
  12. (else (loop (cdr list))))))
  13. (define (check-nulls who the-list the-lists lists)
  14. (for-each (lambda (list)
  15. (if (not (null? list))
  16. (apply assertion-violation who
  17. "argument lists don't have the same size"
  18. list lists)))
  19. lists))
  20. (define (for-all proc list . lists)
  21. (assert-procedure 'for-all proc)
  22. (cond
  23. ((null? lists)
  24. (for-all1 proc list))
  25. ((null? list)
  26. (check-nulls 'for-all list lists lists)
  27. #t)
  28. (else
  29. (let loop ((list list) (lists lists))
  30. (let ((next (cdr list)))
  31. (cond
  32. ((null? next)
  33. (apply proc (car list) (map car lists)))
  34. ((apply proc (car list) (map car lists))
  35. (loop next (map cdr lists)))
  36. (else #f)))))))
  37. (define (for-all1 proc list)
  38. (if (null? list)
  39. #t
  40. (let loop ((list list))
  41. (let ((next (cdr list)))
  42. (cond
  43. ((null? next) (proc (car list)))
  44. ((proc (car list)) (loop next))
  45. (else #f))))))
  46. (define (exists proc list . lists)
  47. (assert-procedure 'exists proc)
  48. (cond
  49. ((null? lists)
  50. (exists1 proc list))
  51. ((null? list)
  52. (check-nulls 'exists list lists lists)
  53. #f)
  54. (else
  55. (let loop ((list list) (lists lists))
  56. (let ((next (cdr list)))
  57. (if (null? next)
  58. (apply proc (car list) (map car lists))
  59. (or (apply proc (car list) (map car lists))
  60. (loop next (map cdr lists)))))))))
  61. (define (exists1 proc list)
  62. (if (null? list)
  63. #f
  64. (let loop ((list list))
  65. (let ((next (cdr list)))
  66. (if (null? next)
  67. (proc (car list))
  68. (or (proc (car list))
  69. (loop next)))))))
  70. (define (filter proc list)
  71. (assert-procedure 'filter proc)
  72. (let loop ((list list) (r '()))
  73. (cond ((null? list)
  74. (reverse r))
  75. ((proc (car list))
  76. (loop (cdr list) (cons (car list) r)))
  77. (else
  78. (loop (cdr list) r)))))
  79. (define (partition proc list)
  80. (assert-procedure 'partition proc)
  81. (let loop ((list list) (yes '()) (no '()))
  82. (cond ((null? list)
  83. (values (reverse yes) (reverse no)))
  84. ((proc (car list))
  85. (loop (cdr list) (cons (car list) yes) no))
  86. (else
  87. (loop (cdr list) yes (cons (car list) no))))))
  88. (define (fold-left combine nil the-list . the-lists)
  89. (assert-procedure 'fold-left combine)
  90. (if (null? the-lists)
  91. (fold-left1 combine nil the-list)
  92. (let loop ((accum nil) (list the-list) (lists the-lists))
  93. (if (null? list)
  94. (begin
  95. (check-nulls 'fold-left the-list the-lists lists)
  96. accum)
  97. (loop (apply combine accum (car list) (map car lists))
  98. (cdr list)
  99. (map cdr lists))))))
  100. (define (fold-left1 combine nil list)
  101. (let loop ((accum nil) (list list))
  102. (if (null? list)
  103. accum
  104. (loop (combine accum (car list))
  105. (cdr list)))))
  106. (define (fold-right combine nil the-list . the-lists)
  107. (assert-procedure 'fold-right combine)
  108. (if (null? the-lists)
  109. (fold-right1 combine nil the-list)
  110. (let recur ((list the-list) (lists the-lists))
  111. (if (null? list)
  112. (begin
  113. (check-nulls 'fold-right the-list the-lists lists)
  114. nil)
  115. (apply combine
  116. (car list)
  117. (append (map car lists)
  118. (cons (recur (cdr list) (map cdr lists))
  119. '())))))))
  120. (define (fold-right1 combine nil list)
  121. (let recur ((list list))
  122. (if (null? list)
  123. nil
  124. (combine (car list) (recur (cdr list))))))
  125. (define (remp proc list)
  126. (assert-procedure 'remp proc)
  127. (let recur ((list list) (res '()))
  128. (cond ((null? list) (reverse res))
  129. ((proc (car list))
  130. (append-reverse! res (recur (cdr list) '())))
  131. (else
  132. (recur (cdr list) (cons (car list) res))))))
  133. ;; Poor man's inliner
  134. (define-syntax define-remove-like
  135. (syntax-rules ()
  136. ((define-remove-like ?name ?equal?)
  137. (define (?name obj list)
  138. (let recur ((list list) (res '()))
  139. (cond ((null? list) (reverse res))
  140. ((?equal? obj (car list))
  141. (append-reverse! res (recur (cdr list) '())))
  142. (else
  143. (recur (cdr list) (cons (car list) res)))))))))
  144. (define-remove-like remove equal?)
  145. (define-remove-like remv eqv?)
  146. (define-remove-like remq eq?)
  147. (define (append-reverse! l1 l2)
  148. (let loop ((list l1) (res l2))
  149. (cond ((null? list)
  150. res)
  151. (else
  152. (let ((next (cdr list)))
  153. (set-cdr! list res)
  154. (loop next list))))))
  155. (define (memp proc list)
  156. (assert-procedure 'member proc)
  157. (let loop ((list list))
  158. (cond ((null? list) #f)
  159. ((proc (car list)) list)
  160. (else (loop (cdr list))))))
  161. (define-syntax define-member-like
  162. (syntax-rules ()
  163. ((define-member-like ?name ?equal?)
  164. (define (?name obj list)
  165. (let loop ((list list))
  166. (cond ((null? list) #f)
  167. ((?equal? obj (car list)) list)
  168. (else (loop (cdr list)))))))))
  169. ; take the versions from `scheme'
  170. ;(define-member-like member equal?)
  171. ;(define-member-like memv eqv?)
  172. ;(define-member-like memq eq?)
  173. (define (assp proc alist)
  174. (assert-procedure 'assp proc)
  175. (let loop ((alist alist))
  176. (if (null? alist)
  177. #f
  178. (let ((p (car alist)))
  179. (if (proc (car p))
  180. p
  181. (loop (cdr alist)))))))
  182. (define-syntax define-assoc-like
  183. (syntax-rules ()
  184. ((define-assoc-like ?name ?equal?)
  185. (define (?name obj alist)
  186. (let loop ((alist alist))
  187. (if (null? alist)
  188. #f
  189. (let ((p (car alist)))
  190. (if (?equal? obj (car p))
  191. p
  192. (loop (cdr alist))))))))))
  193. ; take the versions from `scheme'
  194. ;(define-member-like assoc equal?)
  195. ;(define-member-like assv eqv?)
  196. ;(define-member-like assq eq?)
  197. (define (cons* obj . objs)
  198. (if (null? objs)
  199. obj
  200. (cons obj (apply cons* objs))))