有没有办法用switch语句测试OptionSet?

Gol*_*Joe 7 swift

定义一个简单的OptionSet:

public struct TestSet : OptionSet, Hashable
{
    public let rawValue: Int
    public init(rawValue:Int){ self.rawValue = rawValue}
    public var hashValue: Int {
        return self.rawValue
    }

    public static let A   = TestSet(rawValue: 1 << 0)
    public static let B   = TestSet(rawValue: 1 << 1)
    public static let C   = TestSet(rawValue: 1 << 2)
}
Run Code Online (Sandbox Code Playgroud)

叫它:

let ostest : TestSet = [.A, .B]

switch ostest{
case .A: print("A")
case .B: print("B")
case .C: print("C")
default: print("Default")
}

if ostest.contains(.A){
    print("Contains A")
}
if ostest.contains(.B){
    print("Contains B")
}
Run Code Online (Sandbox Code Playgroud)

输出是:

Default
Contains A
Contains B
Run Code Online (Sandbox Code Playgroud)

有没有办法检查OptionSets是否包含带有switch语句的值或值组合?它比一系列if-contains语句更清晰.

Sco*_*zie 1

您不能直接执行此操作,因为 switch 正在使用 Equatable,并且我认为正在使用 SetAlgebra。

但是,您可以使用以下内容包装 OptionSet:

public struct TestSetEquatable<T: OptionSet>: Equatable {

    let optionSet: T

    public static func == (lhs: Self, rhs: Self) -> Bool {

        return lhs.optionSet.isSuperset(of: rhs.optionSet)
    }
}
Run Code Online (Sandbox Code Playgroud)

这可以让你做:

let ostest : TestSet = [.A, .C]

switch TestSetEquatable(optionSet: ostest) {

  case TestSetEquatable(optionSet: [.A, .B]):
   print("-AB")
   fallthrough

 case TestSetEquatable(optionSet: [.A, .C]):
   print("-AC")
   fallthrough

 case TestSetEquatable(optionSet: [.A]):
   print("-A")
   fallthrough

   default:
     print("-")
}
Run Code Online (Sandbox Code Playgroud)

这打印:

-AC
-A
- // from the fall through to default
Run Code Online (Sandbox Code Playgroud)

意见:我本人不倾向于使用此代码,但如果必须的话,这就是我会做的。