2023-09-01 03:44:46 +00:00
|
|
|
#programming-language #compiler-design
|
|
|
|
|
|
|
|
chumskyのチュートリアルで、評価する関数をライフタイム付きでこんな感じになってたの頭いいなと思ったので、RAIIにしたらもっとシンプルに見えるのではと思った
|
|
|
|
|
|
|
|
```rust
|
|
|
|
fn eval<'a>(expr:Expr,env:&'a mut Vec<(String,Val)>)->Result<Val,Error>{
|
|
|
|
match expr {
|
|
|
|
...
|
|
|
|
Expr::Let(name,e,then)=>{
|
|
|
|
env.push((name,eval(e,env)));
|
|
|
|
let res = eval(then,env);
|
|
|
|
env.pop();
|
|
|
|
res
|
|
|
|
}
|
|
|
|
...
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
Environmentは評価全体で見ればLetやLambdaごとに分岐していく構造だけれど、局所的には1列のベクタで表現できる
|
|
|
|
|
2023-09-01 09:50:32 +00:00
|
|
|
こんな感じすかねえ
|
2023-09-01 03:44:46 +00:00
|
|
|
|
2023-09-01 09:50:32 +00:00
|
|
|
```rust
|
2023-09-01 09:57:37 +00:00
|
|
|
struct EnvironmentT<'a, T: Clone>(&'a mut Vec<(String, T)>, usize);
|
2023-09-01 03:44:46 +00:00
|
|
|
|
2023-09-01 09:57:37 +00:00
|
|
|
impl<'a, T: Clone> EnvironmentT<'a, T> {
|
2023-09-01 09:50:32 +00:00
|
|
|
pub fn new(vec: &'a mut Vec<(String, T)>, mut names: Vec<(String, T)>) -> Self {
|
|
|
|
let len = vec.len();
|
|
|
|
vec.append(&mut names);
|
|
|
|
Self(vec, len)
|
|
|
|
}
|
2023-09-01 09:57:37 +00:00
|
|
|
pub fn lookup(&self, name: &String) -> Option<T> {
|
2023-09-01 09:50:32 +00:00
|
|
|
let res = self
|
|
|
|
.0
|
|
|
|
.iter()
|
|
|
|
.rev()
|
|
|
|
.filter(|(n, _v)| name == n)
|
|
|
|
.collect::<Vec<_>>();
|
2023-09-01 09:57:37 +00:00
|
|
|
res.get(0).map(|(_, v)| v.clone())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, T: Clone> Drop for EnvironmentT<'a, T> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
self.0.truncate(self.1);
|
2023-09-01 09:50:32 +00:00
|
|
|
}
|
2023-09-01 03:44:46 +00:00
|
|
|
}
|
|
|
|
```
|