-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathmain.cpp
72 lines (66 loc) · 1.7 KB
/
main.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
#include "pch.h"
#include <functional>
#include <memory>
#include <type_traits>
template<typename T>
class LazyValue {
private:
struct LazyValueImpl {
mutable function<T()> m_f;
mutable bool m_hasEval;
mutable T m_value;
LazyValueImpl(function<T()> f): m_f(f), m_hasEval(false){}
T& value() const {
if (!m_hasEval) {
m_value = m_f();
m_f = 0;
m_hasEval = true;
}
return m_value;
}
};
public:
LazyValue(){}
LazyValue(function<T()> f): m_impl(make_shared<LazyValueImpl>(f)){ }
T& value() const {
return m_impl->value();
}
private:
shared_ptr<LazyValueImpl> m_impl;
};
template<typename FuncT>
LazyValue<typename result_of<FuncT&()>::type> lazy(const FuncT& f) {
return LazyValue<typename result_of<FuncT&()>::type>(f);
}
//////////////////////////////
int func() {
printf("call %s:%d\n", __func__, __LINE__);
return 3;
}
class HeavyObj {
public:
HeavyObj(){ puts("construct heavy");}
~HeavyObj(){ puts("destruct heavy"); }
int value() { return 10; }
private:
};
int main() {
auto v = lazy(func);
LazyValue<int> v2([v](){
printf("call %s:%d\n", __func__, __LINE__);
return 5 * v.value();
});
LazyValue<int> v3;
{
auto obj = make_shared<HeavyObj>();
v3 = lazy([v, v2, obj](){
printf("call %s:%d\n", __func__, __LINE__);
return v.value() * v2.value() * obj->value();
});
}
puts("--- before");
cout << v3.value() << endl;
puts("--- after");
cout << v3.value() << endl;
cout << v3.value() << endl;
}