1
1
mirror of https://github.com/github/semantic.git synced 2024-12-11 08:45:48 +03:00
semantic/prototype/Doubt/Memo.swift
2015-09-17 14:10:51 -04:00

49 lines
781 B
Swift

public struct Memo<A> {
public init(unevaluted: () -> A) {
self.init(.Unevaluated(unevaluted))
}
public init(evaluated: A) {
self.init(.Evaluated(evaluated))
}
private init(_ thunk: Thunk<A>) {
_value = MutableBox(thunk)
}
public var value: A {
return _value.value.value()
}
private var _value: MutableBox<Thunk<A>>
public func map<B>(transform: A -> B) -> Memo<B> {
return Memo<B> { transform(self.value) }
}
}
private final class MutableBox<A> {
init(_ value: A) {
self.value = value
}
var value: A
}
private enum Thunk<A> {
case Evaluated(A)
case Unevaluated(() -> A)
mutating func value() -> A {
switch self {
case let .Evaluated(a):
return a
case let .Unevaluated(f):
let a = f()
self = .Evaluated(a)
return a
}
}
}