Array with lazy elements

一曲冷凌霜 提交于 2019-12-08 04:26:26

You could use a lazy backing storage for A that is instantiated at first access, e.g.:

class Astorage {
    /* ... lots of ... */
    let id: Int
    var foo: String = "bar"
    init(_ id: Int) {
        self.id = id
        print("Initializing backing storage for A with id \(id)")
        // ...
    }
}

class A {
    private let id: Int
    lazy var storage: Astorage = Astorage(self.id)
    init(_ id: Int) {
        self.id = id
    }
}

class B {
    var array: [A]
    init (_ array: [A]) {
        self.array = array
    }
}

let b = B((1...5).map(A.init))

b.array[2].storage.foo = "foo"
// Initializing backing storage for A with id 3

b.array[4].storage.foo = "bax"
// Initializing backing storage for A with id 5

You can use an enum and a mutating struct for this

enum LazyValueEnum<T>: CustomStringConvertible {
    case value(T)
    case thunk((Void) -> T)

    var forceValue: T {
        switch self {
        case .value(let t):
                return t
        case .thunk(let thunk):
                return thunk()
        }
    }

    var description: String {
        switch self {
        case .value(let t):
            return String(describing: t)
        case .thunk(_):
            return "<thunk>"
        }
    }
}

struct LazyValue<T>: CustomStringConvertible {

    var value: LazyValueEnum<T>

    init(_ t: T) {
        self.value = .value(t)
    }

    init(lazy thunk: @escaping (Void) -> T) {
        self.value = .thunk(thunk)
    }

    static func lazy(_ thunk: @escaping (Void) -> T) -> LazyValue<T> {
        return LazyValue(lazy: thunk)
    }

    static func value(_ value: T) -> LazyValue<T> {
        return LazyValue(value)
    }

    mutating func forceValue() -> T {
        let t = value.forceValue
        value = .value(t)
        return t
    }

    var description: String {
        return value.description
    }
}


var a: [LazyValue<Int>] = [.value(5), .value(7), .lazy { Int(arc4random()) }]
print(a)
print(a[2].forceValue())
print(a)
print(a[2].forceValue())
print(a)

Results from a playground:

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!