[swift-users] Type "T?" does not conform to protocol 'Equatable'

davesweeris at mac.com davesweeris at mac.com
Mon Feb 29 10:41:43 CST 2016


To the best of my understanding, Int? is not equatable. Your let c = a == b  code works because this function (or something like it) is defined:
func == <T: Equatable> (lhs: Optional<T>, rhs: Optional<T>) -> Bool {
    switch (lhs, rhs) {
    case (.None, .None): return true
    case (.Some(let l), .Some(let r)): return l == r
    default: return false
    }
}

The reason it’s done this way is that currently there’s no mechanism for conditional protocol conformance. Ideally, you’d be able to write something like this:
extension Optional : Equatable where T: Equatable {}
… which, combined with that existing == function, would make Int? be equatable.

IIRC, Apple is intending to remove this limitation, but I can’t remember if it’s made the cut for Swift 3 or if it’ll be later on.

In the meantime, if Keith’s solution of making the variable be a T? instead of T doesn’t work for you, I think this will let you continue on with other parts of your code (although I have NOT thoroughly tested it… use at your own risk):
public struct OptionalWrapper<T> : Equatable {
    typealias Wrapped = T
    var unwrap: Wrapped?
    init(_ value: Wrapped?) { unwrap = value }
}
public func == <T: Equatable> (lhs: OptionalWrapper<T>, rhs: OptionalWrapper<T>) -> Bool {
    switch (lhs.unwrap, rhs.unwrap) {
    case (.None, .None): return true
    case (.Some(let l), .Some(let r)): return l == r
    default: return false
    }
}
public func == <T: Equatable> (lhs: OptionalWrapper<T?>, rhs: OptionalWrapper<T?>) -> Bool {
    switch (lhs.unwrap, rhs.unwrap) {
    case (.None, .None): return true
    case (.Some(let l), .Some(let r)): return l == r
    default: return false
    }
}
public func == <T> (lhs: OptionalWrapper<T>, rhs: OptionalWrapper<T>) -> Bool {
    switch (lhs.unwrap, rhs.unwrap) {
    case (.None, .None): return true
    case (.Some, .None): return false
    case (.None, .Some): return false
    case (.Some, .Some): fatalError("\(__FILE__):\(__LINE__):\(__COLUMN__): Binary operator '==' cannot be applied to two '\(T.self)' operands")
    }
}

The syntax for actually using it is a bit long, but it should work: 
let a = C(t: OptionalWrapper(4)) // infers that Wrapped == Int
let b = C(t: OptionalWrapper<Int>(nil)) // must be explicit since nil is just Optional.None, without any other type info
let c = a.t == b.t // false

Hope that helps

- Dave Sweeris

> On Feb 29, 2016, at 8:24 AM, Rudolf Adamkovič via swift-users <swift-users at swift.org> wrote:
> 
> Hi everyone!
> 
> I have a generic class similar to this one:
> 
> class C<T: Equatable> {
>     let t: T
>     init(t: T) { self.t = t }
> }
> 
> When try to wrap ‘Int?' inside, I get the following error:
> 
> let a = C<Int?>(t: nil) // ERROR: Type "Int?" does not conform to protocol 'Equatable'
> 
> Yet when I try to compare two ‘Int?’ values, everything works:
> 
> let a: Int? = 5
> let b: Int? = 6
> 
> let c = a == b // NO ERROR
> 
> So, is ‘Int?' equatable or not?
> 
> Thanks!
> 
> R+
> _______________________________________________
> swift-users mailing list
> swift-users at swift.org
> https://lists.swift.org/mailman/listinfo/swift-users

-------------- next part --------------
An HTML attachment was scrubbed...
URL: <https://lists.swift.org/pipermail/swift-users/attachments/20160229/65e0e88d/attachment.html>


More information about the swift-users mailing list