-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathTest.cpp
148 lines (131 loc) · 3.07 KB
/
Test.cpp
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
struct True {
static const int value = 1;
};
struct False {
static const int value = 0;
};
template<bool, typename Conseq, typename Alt>
struct If {
typedef Conseq Result;
};
template<typename Conseq, typename Alt>
struct If<false, Conseq, Alt> {
typedef Alt Result;
};
struct Nil{};
template<typename Car, typename Cdr>
struct Cons {
typedef Car Car;
typedef Cdr Cdr;
};
template<typename List, typename Key>
struct Assq {
typedef typename If<
List::Car::Car::value == Key::value,
typename List::Car,
typename Assq<typename List::Cdr, Key>::Result>::Result Result;
};
template<typename Key>
struct Assq<Nil, Key> {
typedef False Result;
};
template<typename List, typename Key>
struct Lookup {
typedef typename Assq<List, Key>::Result::Cdr Result;
};
template<typename Expr, typename Env>
struct Eval;
template<int _value>
struct Const {
static const int value = _value;
};
template<int _value>
struct Var {
static const int value = _value;
};
template<typename Formal, typename Body>
struct Lambda {
typedef Formal Formal;
typedef Body Body;
};
template<typename Func, typename Actual>
struct Application {
typedef Func Func;
typedef Actual Actual;
};
template<typename Lambda, typename Env>
struct Closure {
template<typename Actual>
struct Apply {
typedef typename Eval<typename Lambda::Body, Cons<Cons<typename Lambda::Formal, Actual>, Env>>::Result Result;
};
};
template<int value, typename Env>
struct Eval<Const<value>, Env> {
typedef Const<value> Result;
};
template<int n, typename Env>
struct Eval<Var<n>, Env> {
typedef typename Lookup<Env, Var<n>>::Result Result;
};
template<typename Formal, typename Body, typename Env>
struct Eval<Lambda<Formal, Body>, Env> {
typedef Closure<Lambda<Formal, Body>, Env> Result;
};
template<typename Func, typename Actual, typename Env>
struct Eval<Application<Func, Actual>, Env> {
typedef typename Eval<Func, Env>::Result FuncResult;
typedef typename Eval<Actual, Env>::Result ActualResult;
typedef typename FuncResult::template Apply<ActualResult>::Result Result;
};
struct Add {
template<typename T1>
struct Apply {
struct Result {
template<typename T2>
struct Apply {
typedef Const<T1::value + T2::value> Result;
};
};
};
};
int main() {
typedef Var<0> x;
typedef Var<1> y;
typedef Var<2> z;
typedef Var<3> add;
typedef Cons<Cons<add, Add>, Nil> InitialEnv;
cout
<< "((lambda (x) (+ 1 x)) 2)"
<< " ==> "
<< Eval<
Application<
Lambda<x,
Application<Application<add, Const<1>>, x>>,
Const<2>>,
InitialEnv>::Result::value
<< endl;
cout
<< "((lambda (x) (+ (+ 1 x) (+ 2 x))) 3)"
<< " ==> "
<< Eval<
Application<
Lambda<x,
Application<
Application<add, Application<Application<add, x>, Const<1>>>,
Application<Application<add, x>, Const<2>>>>,
Const<3>>,
InitialEnv>::Result::value
<< endl;
cout
<< "(((lambda (x) (lambda (y) (+ x y))) 2) 3)"
<< " ==> "
<< Eval<
Application<
Application<
Lambda<x, Lambda<y, Application<Application<add, x>, y>>>,
Const<2>>,
Const<3>>,
InitialEnv>::Result::value
<< endl;
}