-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathanaphoric-op.lisp
146 lines (117 loc) · 4.01 KB
/
anaphoric-op.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
(in-package :arc-compat.internal)
(in-readtable :common-lisp)
(in-suite arc-compat)
;;; '(w/uniq)
;;; Copyright 1995 by Paul Graham.
;;; http://lib.store.yahoo.net/lib/paulgraham/onlisp.lisp
;;; http://lib.store.yahoo.net/lib/paulgraham/utx.lisp
;;; http://lib.store.yahoo.net/lib/paulgraham/acl2.lisp
;;;
;;; Documentation String
;;; http://arcfn.com/doc/
;;; Copyright 2008 Ken Shirriff.
;;; Anaphoric operations
;FIXME
;[code] [Macro] aif expr body [expr body] ...
#|(defmacro aif (test-form then-form &optional else-form)
"Anaphoric if:
each expr is evaluated until one is true, and then the corresponding body is executed.
Within the body, the anaphoric variable it refers back to the value of expr."
`(cl:let ((it ,test-form))
(if it ,then-form ,else-form)))|#
;>
;(aif (> 1 2) (+ it 1) 42 (+ it 2))
;44
;>(aif nil (+ it 1))
;nil
(mac iflet (var expr then . rest)
(w/uniq gv
`(let ,gv ,expr
(if ,gv (let ,var ,gv ,then) ,@rest))))
;[code] [Macro] awhen expr [body ...]
;<- *onlisp*
(defmacro awhen (test-form &body body)
"Anaphoric when: if the expr is true, the body is executed.
Within the body, the variable it refers back to the value of expr."
`(aif ,test-form
(progn ,@body)))
;>(awhen (* 2 3) (+ it 1))
;7
(mac whenlet (var expr &body body)
`(iflet ,var ,expr (do ,@body)))
(mac unless (test . body)
`(if (no ,test) (do ,@body)))
(defmacro afn (params &body body)
"Creates an anaphoric function, which can be called recursively
with the name self. This allows a recursive function to be
created without assigning it a name."
(cl:let ((args (gensym)))
`(labels ((self (&rest ,args)
(cl:declare (cl:dynamic-extent ,args))
(destructuring-bind ,params ,args
(cl:declare (cl:ignorable
,@(remove-if (lambda (x)
(member x cl:lambda-list-keywords))
(+internal-flatten params))))
,@body)))
#'self)))
(tst afn
(== (funcall (afn (x) (if (is x 0) 1 (* 2 (self (- x 1))))) 5)
32))
(mac caselet (var expr . args)
(let ex (afn (args)
(if (cl:null (cdr args))
(car args)
`(if (is ,var ',(car args))
,(cadr args)
,(self (cddr args)))))
`(let ,var ,expr ,(funcall ex args))))
(mac case (expr . args)
`(caselet ,(uniq) ,expr ,@args))
(mac check (x test (o alt))
(w/uniq gx
`(let ,gx ,x
(if (,test ,gx) ,gx ,alt))))
;[code] [Macro] aand [arg ...]
(defmacro aand (&rest args)
"Anaphoric and. Returns the last argument if all arguments are
true, otherwise returns nil. Inside each argument the anaphoric
variable it refers to the value of the previous argument. Like
and, lazy evaluation is used, so evaluation stops after
encountering a false argument."
(cond ((null args) t)
((null (cdr args)) (car args))
(t `(if ,(car args) (and ,@(cdr args))))))
;>(aand 1 (+ it 2) (* it 10))
;30
;FIXME
;[code] [Macro] rfn name parms [body ...]
(defmacro rfn (name parms &body body)
"Creates a function with the given name. The name is only inside
the scope of the rfn macro. This allows recursive functions to be
created without polluting the wider scope."
`(labels ((,name ,parms ,@body))
(cl:declare
(cl:optimize (cl:speed 3) (cl:debug 0) (cl:space 0)
(cl:safety 1)))
#|(cl:declare (cl:optimize (cl:debug 1)))|#
#',name))
;(funcall (rfn pow2 (x) (if (= x 0) 1 (* 2 (pow2 (- x 1))))) 5)
; 32
;[code] [Procedure] trav obj [function ...]
(defmacro trav (x &rest fs)
"Recursive traversal. Applies each function in sequence to obj,
if obj is not nil. The function can recursively call itself with
a new obj with (self obj)."
(w/uniq g
`(labels ((self (,g)
(when ,g
,@(mapcar (lambda (f) `(,f ,g)) fs))))
(self ,x))))
;(trav '(1 2 3 4) (lambda (_) (print _)) (lambda (_) (self (cdr _))))
;
;(1 2 3 4)
;(2 3 4)
;(3 4)
;(4)
;nil