Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Swift Covariant Generics

Here is an example of what I'd like to achieve:

protocol SomeType {}

class SomeClass: SomeType {}

struct SomeGenericStruct<A> {
    typealias E = A
}

func take(someType: SomeGenericStruct<SomeType>) {}

let concreteGenericStruct1: SomeGenericStruct<SomeType> = SomeGenericStruct<SomeType>()
let concreteGenericStruct2: SomeGenericStruct<SomeClass> = SomeGenericStruct<SomeClass>()

take(concreteGenericStruct1)
take(concreteGenericStruct2) // much no work, very repair. wow.

Or even simpler:

let concreteGenericStruct3: SomeGenericStruct<SomeType> = SomeGenericStruct<SomeClass>() as SomeGenericStruct<SomeType> // still no work

How can I manage to provide take with concreteGenericStruct2?

like image 398
fabb Avatar asked Mar 08 '16 08:03

fabb


1 Answers

You can use generic method for this:

func take<T where T: SomeType>(someType: SomeGenericStruct<T>) { }

The only problem with this is that you can not pass SomeGenericStruct<SomeType> to it. It must be a generic of some concrete type instead. If totally necessary, you can just have two functions doing the same thing essentially:

func take(someInput: SomeGenericStruct<SomeType>) { /* do stuff */ }
func take<T where T: SomeType>(someType: SomeGenericStruct<T>) { /* do same stuff */ }
like image 136
0x416e746f6e Avatar answered Sep 27 '22 22:09

0x416e746f6e