-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtree.rb
294 lines (244 loc) · 5.91 KB
/
tree.rb
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
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
# Lambda Expression Tree
class LambdaExpressionNode
# todo: implements variables method
def initialize
@mark = false
end
def mark_first_redex(canonical = false)
return false
end
def reduce(e = [], q = [])
return nil
end
def beta(i)
method = self.method("beta#{i}".intern)
return method.call
end
def variables(binding = [])
return []
end
end
class VariableNode < LambdaExpressionNode
def initialize(var)
@var = var
end
def variables(binding = [])
if binding.include?(self) then
return []
else
return [self]
end
end
def ==(target)
return false unless target.instance_of?(VariableNode)
return self.to_s == target.to_s
end
def to_s
return @var
end
end
class ConstantNode < LambdaExpressionNode
attr_accessor :num
def initialize(num)
@num = num
end
def variables(binding = [])
return []
end
def ==(target)
self.num == target.num
end
def to_s
return @num.to_s
end
end
class ApplicationNode < LambdaExpressionNode
attr_accessor :expr1, :expr2, :mark
def initialize(expr1, expr2)
@expr1 = expr1
@expr2 = expr2
end
def mark_first_redex(canonical = false)
for i in 1..5 do
method = self.method("beta#{i}?".intern)
@mark = "beta#{i}" if method.call
return true if @mark
end
return true if (alpha? and alpha)
mark1 = @expr1.mark_first_redex(canonical)
return true if mark1
mark2 = @expr2.mark_first_redex(canonical)
return true if mark2
return false
end
def reduce(e = [], q = [])
if @mark then
return beta1 if beta1?
return beta2 if beta2?
return beta3 if beta3?
return beta4, true if beta4?
return beta5 if beta5?
end
if @expr1.instance_of?(AbstractionNode) and
@expr1.expr.instance_of?(AbstractionNode) then
e << @expr1.expr.expr.variables
q << @expr2.variables
end
node, marked = @expr1.reduce(e, q)
if node then
@expr1 = node
return self, marked
end
node, marked = @expr2.reduce(e, q)
if node then
@expr2 = node
return self, marked
end
return self
end
def variables(binding = [])
first_vars = @expr1.variables(binding)
second_vars = @expr2.variables(binding)
return first_vars + second_vars
end
# test alpha reduce(based on beta4?)
def alpha?
if @expr1.instance_of?(AbstractionNode) and
@expr1.expr.instance_of?(AbstractionNode) and
@expr1.var != @expr1.expr.var and
@expr1.expr.expr.variables.include?(@expr1.var) and
@expr2.variables.include?(@expr1.expr.var) then
return true
end
return false
end
# mark
def alpha
return nil unless alpha?
@expr1.expr.mark = true
return self
end
def beta1?
if @expr1.instance_of?(AbstractionNode) and
@expr1.expr.instance_of?(VariableNode) and
@expr1.var == @expr1.expr then
return true
end
return false
end
def beta1
return nil unless beta1?
return @expr2 if beta1?
end
def beta2?
if @expr1.instance_of?(AbstractionNode) and
@expr1.expr.instance_of?(VariableNode) and
@expr1.var != @expr1.expr then
return true
end
return false
end
def beta2
return nil unless beta2?
return @expr1.expr if beta2?
end
def beta3?
if @expr1.instance_of?(AbstractionNode) and
@expr1.expr.instance_of?(AbstractionNode) and
@expr1.var == @expr1.expr.var then
return true
end
return false
end
def beta3
return nil unless beta3?
return AbstractionNode.new(@expr1.var, @expr1.expr.expr)
end
def beta4?
if @expr1.instance_of?(AbstractionNode) and
@expr1.expr.instance_of?(AbstractionNode) and
@expr1.var != @expr1.expr.var and
( not @expr1.expr.expr.variables.include?(@expr1.var) or
not @expr2.variables.include?(@expr1.expr.var) ) then
return true
end
return false
end
def beta4
return nil unless beta4?
x = AbstractionNode.new(@expr1.var, @expr1.expr.expr)
q = ApplicationNode.new(x, @expr2)
q.mark = true
return AbstractionNode.new(@expr1.expr.var, q)
end
def beta5?
if @expr1.instance_of?(AbstractionNode) and
@expr1.expr.instance_of?(ApplicationNode) then
return true
end
return false
end
def beta5
return nil unless beta5?
e1 = AbstractionNode.new(@expr1.var, @expr1.expr.expr1)
e2 = AbstractionNode.new(@expr1.var, @expr1.expr.expr2)
ee1 = ApplicationNode.new(e1, @expr2)
ee2 = ApplicationNode.new(e2, @expr2)
return ApplicationNode.new(ee1, ee2)
end
def to_s
return "(#{@expr1.to_s})#{@expr2.to_s}"
end
end
class AbstractionNode < LambdaExpressionNode
attr_accessor :var, :expr, :mark
def initialize(var, expr)
@var = var
@expr = expr
end
def mark_first_redex(canonical = false)
return @expr.mark_first_redex(canonical)
end
def variables(binding = [])
binding << @var
return @var.variables(binding) + @expr.variables(binding)
end
# e and q are bindings
def reduce(e = [], q = [])
if @mark then
return alpha(e, q) if alpha?(e, q)
end
node, marked = @expr.reduce
if node then
@expr = node
return self, marked
end
return self
end
def alpha?(e, q)
var = nil
for z in 'a'..'z' do
var = VariableNode.new(z)
break unless e.include?(var)
break unless q.include?(var)
var = nil
end
return true if var
end
def alpha(e, q)
var = nil
for z in 'a'..'z' do
var = VariableNode.new(z)
break unless e.include?(var)
break unless q.include?(var)
var = nil
end
raise "No More Free Variable" unless var
app = ApplicationNode.new(AbstractionNode.new(@var, @expr), var)
app.mark = true
return AbstractionNode.new(var, app), true
end
def to_s
return "&#{@var}.#{@expr}"
end
end