2
class MyClass 
{
    enum MyEnum {
        case FirstCase
        case SecondCase(Int)
        case ThirdCase
    }

    var state:MyEnum!

    func myMethod ()
    {
        if state! == MyEnum.FirstCase {
            // Do something
        }
    }
}

我得到指向if语句的编译器错误::

二元运算符“==”不能应用于两个“MyClass.MyEnum”操作数

如果相反,我使用一个switch语句,则没有问题:

switch state! {
    // Also, why do I need `!` if state is already an 
    // implicitly unwrapped optional? Is it because optionals also 
    // are internally enums, and the compiler gets confused?

case .FirstCase:
    // do something...

default:
    // (do nothing)
    break
}

然而,这switch句话感觉太冗长了:我只想do somethingfor .FirstCase,没有别的。if声明更有意义。

枚举和发生了==什么?

编辑:这太奇怪了。在确定switch版本并转到我的代码的其他(完全不相关的)部分并返回后,if-statement 版本(将强制展开的属性与固定的枚举大小写进行比较)正在编译,没有错误。我只能得出结论,这与解析器中的一些损坏的缓存有关,这些缓存在此过程中被清除了。

编辑2 (感谢@LeoDabus 和@MartinR):当我将关联值设置为另一个枚举案例(不是我要比较的那个 - 在这种情况下为.SecondCase)时,似乎出现了错误。我仍然不明白为什么会特别触发这个编译器错误(“不能使用二元运算符'=='...”),或者这意味着什么。

4

2 回答 2

8

正如您在评论中所说,您的枚举类型实际上具有关联的值。==在这种情况下,枚举类型没有默认运算符。

但是你甚至可以在语句中使用模式匹配if(从 Swift 2 开始):

class MyClass {
    enum MyEnum {
        case FirstCase
        case SecondCase
        case ThirdCase(Int)
    }

    var state:MyEnum!

    func myMethod () {
        if case .FirstCase? = state {

        }
    }
}

.FirstCase?.Some(MyEnum.FirstCase).

在您的 switch 语句中state,即使它是隐式展开的可选选项,它也不会自动展开(否则您无法匹配反对nil)。但是在这里可以使用相同的模式:

switch state {
case .FirstCase?:
    // do something...
default:
    break
}

更新:Swift 4.1 (Xcode 9.3) 开始,编译器可以为具有关联值的枚举综合符合 Equatable/Hashable(如果它们的所有类型都是 Equatable/Hashable)。声明一致性就足够了:

class MyClass {
    enum MyEnum: Equatable {
        case firstCase
        case secondCase
        case thirdCase(Int)
    }

    var state:MyEnum!

    func myMethod () {
        if state  == .firstCase {
            // ...
        }
    }
}
于 2015-11-06T06:36:33.583 回答
2
class MyClass {
    enum MyEnum {
        case FirstCase
        case SecondCase
        case ThirdCase
    }

    var state: MyEnum!

    func myMethod()  {
        guard let state = state else { return }
        if state == MyEnum.FirstCase {
            // Do something
            print(true)
        } else {
             print(false)
        }
    }
}

let myClass = MyClass()
myClass.state = .FirstCase
myClass.myMethod()
myClass.state = .SecondCase
myClass.myMethod()
于 2015-11-06T06:16:28.243 回答