about summary refs log tree commit diff stats
path: root/js/games/nluqo.github.io/~bh/downloads/csls-programs/streams
blob: 3fc96a9200264646c6b987ac6b6460b1ce3dfca5 (plain) (blame)
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
; Implementation of SICP streams (lazy-evaluation lists) in Logo.

; Since we don't have special forms, we put the second argument to STREAM
; in a (quoted) list.

; Since we don't have lexical scope, we use substitution (`) into saved
; expressions.

to stream :car :delayed.cdr
output fput :car (list "*delayed* :delayed.cdr)
end

to head :stream
output first :stream
end

to tail :stream
if emptyp bf :stream [output []]
if not equalp first bf :stream "*delayed* [output bf :stream]
localmake "result run last :stream
.setbf :stream :result
output :result
end

; higher order functions for streams

; Remember that if the functional argument uses local variables, it has to
; be backquoted.

to stream.map :fun [:streams] 2
if emptyp first :streams [output []]
output stream apply :fun firsts :streams ~
           `[(apply "stream.map fput ,[quoted :fun] (map "tail ,[:streams]))]
end

to stream.filter :fun :stream
if emptyp :stream [output []]
if invoke :fun head :stream ~
  [output stream head :stream `[stream.filter ,[quoted :fun] tail ,[:stream]]]
output stream.filter :fun tail :stream
end

to flatten :stream.of.streams
if emptyp :stream.of.streams [output []]
output flatten1 head :stream.of.streams :stream.of.streams
end

to flatten1 :stream :delayed.more.streams
if emptyp :stream [output flatten tail :delayed.more.streams]
output stream (head :stream) ~
              `[flatten1 tail ,[:stream]
                         ,[:delayed.more.streams]]
end

; helper for debugging

to show.stream :stream [:num 10]
show show.stream1 :stream :num
end

to show.stream1 :stream :num
if emptyp :stream [output []]
if equalp :num 0 [output [...]]
output fput head :stream (show.stream1 tail :stream :num-1)
end

; examples

to integers.from :n
output stream :n `[integers.from ,[:n+1]]
end

make "integers integers.from 1

to sieve :stream
output stream (head :stream) ~
              `[sieve stream.filter [not divisiblep ? ,[head :stream]]
                                    tail ,[:stream]]
end

to divisiblep :big :small
output 0 = remainder :big :small
end

make "primes sieve tail :integers