1
1
mirror of https://github.com/github/semantic.git synced 2025-01-02 20:41:38 +03:00
semantic/prototype/Doubt/Memo.swift
2015-09-25 11:57:28 -04:00

62 lines
1.1 KiB
Swift

public struct Memo<A>: CustomDebugStringConvertible, CustomStringConvertible {
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) }
}
public func flatMap<B>(transform: A -> Memo<B>) -> Memo<B> {
return Memo<B> { transform(self.value).value }
}
public var description: String {
return String(value)
}
public var debugDescription: String {
return String(reflecting: 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
}
}
}