【问题标题】:Compiler error when comparing values of enum type with associated values?将枚举类型的值与关联值进行比较时出现编译器错误?
【发布时间】:2016-02-07 04:41:21
【问题描述】:
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 something 代替.FirstCase,除此之外别无他法。 if 语句更有意义。

枚举和== 是怎么回事?

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

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

【问题讨论】:

  • 您忘记初始化状态并且您正在强制解开它。添加保护让状态=状态其他{返回}而不是
  • 这是类型和变量名已更改的虚拟代码,大部分方法被省略。在我的真实代码中,var is 已初始化。无论哪种情况,这最多应该是一个运行时问题。其实我在测试viewDidLoad()里面的值,编译器不知道有没有初始化。
  • 您使用的是哪个 Xcode 版本?我在这里没有收到此错误
  • 那是你的真实代码吗?正如 Leo 所说,它在 Xcode 7 和 7.1 中编译没有问题。 – 也许你有一个带有关联值的枚举?
  • @NicolasMiari:另一种选择是让您的枚举符合Equatable,如下所示:stackoverflow.com/a/25726677/59541

标签: swift if-statement enums binary-operators


【解决方案1】:
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()

【讨论】:

  • 如果不是先用guard 解包,而是跳过它并尝试比较state! == MyEnum.FirstCase,会发生什么?
  • @NicolasMiari 只要您初始化 state 属性,您的代码就可以在这里正常运行
  • 我再说一遍:编译器(静态分析器?)无法知道属性是否在使用位置被初始化(viewDidLoad),我正在强制展开它(我不需要这样做,因为它是一个隐式展开的可选选项)。我得到的编译器错误与初始化无关,而是比较两个枚举相同类型(?????)
【解决方案2】:

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

但您甚至可以在 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 {
            // ...
        }
    }
}

【讨论】:

  • 非常感谢。我最终在当前代码中不需要关联值,因此 if 语句按最初计划工作,但会记住此模式以供将来参考。
  • @Martin R: Optional(E.A) == E.A // 在 Swift 2 中给了我真实的感觉
  • @user3441734:抱歉,我不明白你在问什么。
  • 有什么方法可以在一行中做到这一点?例如。 let isEqual: Bool = .FirstCase == state
  • 由于SE-0185,从Swift 4.1 开始,Swift 还支持为具有关联值的枚举合成EquatableHashable
猜你喜欢
  • 2011-11-09
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-10-14
  • 2011-05-28
  • 2017-03-17
  • 1970-01-01
相关资源
最近更新 更多