Search code examples
swiftswift5swift-protocolsassociated-types

Swift: Set protocol's associated type in argument / member variable


I'm trying to create a generic protocol that can be reused in various parts throughout my application. Previously I've hardcoded KeyValueStore to only use String and Int for Key and Value respectively. I ran into an issue where I want a different consumer, class ConsumerB, to use KeyValueStore with Key/Value being Int/Int respectively.

protocol KeyValueStore {
    associatedtype Key
    associatedtype Value
    func get(key: Key) -> Value
    func set(key: Key, value: Value)
}

For the original implementation, I had two classes the conform to the protocol. UserDefaultsStore was used to store data during runtime and MemoryKeyValueStore was used in unit testing.

class UserDefaultsStore: KeyValueStore {
    func get(key: String) -> Int { return 0
        // Implementation
    }
    
    func set(key: String, value: Int) {
        // Implementation
    }
}

class MemoryKeyValueStore: KeyValueStore {
    var values: [String: Int] = [:]
    func get(key: String) -> Int { return values[key]! }
    func set(key: String, value: Int) { values[key] = value}
}

Since they both used KeyValueStore with hardcoded types I could provide a store implementation at runtime.

class ConsumerA {
    var keyValueStore: KeyValueStore
    
    init(store: KeyValueStore) {
        keyValueStore = store
    }
  
    func example() {
        keyValueStore.set(key: "Hello", value: 5)
    }
}

After adding the associated types, I get the error below. It makes sense that while compiling ConsumerA , there is no way to know what type Key and Value take on in the KeyValueStore argument and member variable. However after poking around, I could not seem to find a clear answer on how to properly declare what types Key and Value should be in the ConsumerA class.

In the example function, it is clear it would use String and Int, but how do I set the argument/member variable to only allow classes that conform to KeyValueStore and have Key/Value set to String/Int?

Protocol 'KeyValueStore' can only be used as a generic constraint because it has Self or associated type requirements

Edit: I ended up going a different route based on this medium article I found later in the evening. While it seems like Swift should just natively provide a way to declare a protocol with associated types as a argument/variable type here is the work around.

class AnyKeyValueStore<Key, Value>: KeyValueStore {
    private let _set: (Value?, Key) -> Void
    private let _object: (Key) -> Value?
    
    init<U: KeyValueStore>(_ keyValueStore: U) where U.Key == Key, U.Value == Value {
        _set = keyValueStore.set
        _object = keyValueStore.object
    }
    
    func set(value: Value?, forKey key: Key) {
        _set(value, key)
    }
    
    func object(forKey key: Key) -> Value? {
        return _object(key)
    }
}

Solution

  • ConsumerA needs to be generic. Replace ModuleName with your actual module name.

    class ConsumerA<KeyValueStore: ModuleName.KeyValueStore>
    where KeyValueStore.Key == String, KeyValueStore.Value == Int {
    

    Also, your method pair should be a subscript instead.

    subscript(key: Key) -> Value { get set }
    
    keyValueStore["Hello"] = 5