mirror of
https://github.com/github/semantic.git
synced 2024-11-28 10:15:55 +03:00
62 lines
1.1 KiB
Swift
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
|
|
}
|
|
}
|
|
}
|