亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

如何在Swift中創建NS_OPTIONS樣式的位掩碼枚舉?

如何在Swift中創建NS_OPTIONS樣式的位掩碼枚舉?

墨色風雨 2019-10-05 15:30:55
在Apple有關與C API交互的文檔中,他們描述了將NS_ENUM標記的C樣式枚舉作為Swift枚舉導入的方式。這是有道理的,并且由于Swift中的枚舉很容易作為enum值類型提供,因此很容易看到如何創建我們自己的枚舉。再往下,它說了關于NS_OPTIONS標記C樣式的選項:Swift還會導入標有NS_OPTIONS宏的選項。而選項的行為類似于進口枚舉,選項還可以支持一些位操作,如&,|和~。在Objective-C中,您表示一個常量為零(0)的空選項集。在Swift中,用于nil表示沒有任何選項。鑒于optionsSwift中沒有值類型,我們如何創建要使用的C-Style選項變量?
查看完整描述

3 回答

?
隔江千里

TA貢獻1906條經驗 獲得超10個贊

斯威夫特3.0

與Swift 2.0幾乎相同。OptionSetType重命名為OptionSet,并且按慣例將枚舉寫為小寫。


struct MyOptions : OptionSet {

    let rawValue: Int


    static let firstOption  = MyOptions(rawValue: 1 << 0)

    static let secondOption = MyOptions(rawValue: 1 << 1)

    static let thirdOption  = MyOptions(rawValue: 1 << 2)

}

noneSwift 3建議不要提供選項,而只是使用空數組文字:


let noOptions: MyOptions = []

其他用法:


let singleOption = MyOptions.firstOption

let multipleOptions: MyOptions = [.firstOption, .secondOption]

if multipleOptions.contains(.secondOption) {

    print("multipleOptions has SecondOption")

}

let allOptions = MyOptions(rawValue: 7)

if allOptions.contains(.thirdOption) {

    print("allOptions has ThirdOption")

}

斯威夫特2.0

在Swift 2.0中,協議擴展會處理大多數樣板文件,這些樣板文件現在已作為符合的結構導入OptionSetType。(RawOptionSetType自Swift 2 beta 2起已消失。)聲明要簡單得多:


struct MyOptions : OptionSetType {

    let rawValue: Int


    static let None         = MyOptions(rawValue: 0)

    static let FirstOption  = MyOptions(rawValue: 1 << 0)

    static let SecondOption = MyOptions(rawValue: 1 << 1)

    static let ThirdOption  = MyOptions(rawValue: 1 << 2)

}

現在我們可以使用基于集合的語義MyOptions:


let singleOption = MyOptions.FirstOption

let multipleOptions: MyOptions = [.FirstOption, .SecondOption]

if multipleOptions.contains(.SecondOption) {

    print("multipleOptions has SecondOption")

}

let allOptions = MyOptions(rawValue: 7)

if allOptions.contains(.ThirdOption) {

    print("allOptions has ThirdOption")

}

斯威夫特1.2

看著由夫特導入的(Objective-C的選項UIViewAutoresizing,例如),我們可以看到,選項被聲明為struct符合協議RawOptionSetType,這反過來又符合  _RawOptionSetType,Equatable,RawRepresentable,BitwiseOperationsType,和NilLiteralConvertible。我們可以這樣創建自己的:


struct MyOptions : RawOptionSetType {

    typealias RawValue = UInt

    private var value: UInt = 0

    init(_ value: UInt) { self.value = value }

    init(rawValue value: UInt) { self.value = value }

    init(nilLiteral: ()) { self.value = 0 }

    static var allZeros: MyOptions { return self(0) }

    static func fromMask(raw: UInt) -> MyOptions { return self(raw) }

    var rawValue: UInt { return self.value }


    static var None: MyOptions { return self(0) }

    static var FirstOption: MyOptions   { return self(1 << 0) }

    static var SecondOption: MyOptions  { return self(1 << 1) }

    static var ThirdOption: MyOptions   { return self(1 << 2) }

}

現在,我們可以像對待MyOptions蘋果文檔中所述的那樣對待這個新的選項集:您可以使用enum-like語法:


let opt1 = MyOptions.FirstOption

let opt2: MyOptions = .SecondOption

let opt3 = MyOptions(4)

而且它的行為也像我們期望的選項那樣:


let singleOption = MyOptions.FirstOption

let multipleOptions: MyOptions = singleOption | .SecondOption

if multipleOptions & .SecondOption != nil {     // see note

    println("multipleOptions has SecondOption")

}

let allOptions = MyOptions.fromMask(7)   // aka .fromMask(0b111)

if allOptions & .ThirdOption != nil {

    println("allOptions has ThirdOption")

}

我已經構建了一個生成器來創建Swift選項集,而無需進行所有查找/替換。


最新:對Swift 1.1 beta 3的修改。


查看完整回答
反對 回復 2019-10-05
?
守候你守候我

TA貢獻1802條經驗 獲得超10個贊

Xcode 6.1 Beta 2對RawOptionSetType協議進行了一些更改(請參閱此Airspeedvelocity博客條目和Apple發行說明)。


基于Nate Cooks的示例,這里是更新的解決方案。您可以這樣定義自己的選項集:


struct MyOptions : RawOptionSetType, BooleanType {

    private var value: UInt

    init(_ rawValue: UInt) { self.value = rawValue }


    // MARK: _RawOptionSetType

    init(rawValue: UInt) { self.value = rawValue }


    // MARK: NilLiteralConvertible

    init(nilLiteral: ()) { self.value = 0}


    // MARK: RawRepresentable

    var rawValue: UInt { return self.value }


    // MARK: BooleanType

    var boolValue: Bool { return self.value != 0 }


    // MARK: BitwiseOperationsType

    static var allZeros: MyOptions { return self(0) }


    // MARK: User defined bit values

    static var None: MyOptions          { return self(0) }

    static var FirstOption: MyOptions   { return self(1 << 0) }

    static var SecondOption: MyOptions  { return self(1 << 1) }

    static var ThirdOption: MyOptions   { return self(1 << 2) }

    static var All: MyOptions           { return self(0b111) }

}

然后可以像這樣使用它來定義變量:


let opt1 = MyOptions.FirstOption

let opt2:MyOptions = .SecondOption

let opt3 = MyOptions(4)

像這樣測試位:


let singleOption = MyOptions.FirstOption

let multipleOptions: MyOptions = singleOption | .SecondOption

if multipleOptions & .SecondOption {

    println("multipleOptions has SecondOption")

}


let allOptions = MyOptions.All

if allOptions & .ThirdOption {

    println("allOptions has ThirdOption")

}


查看完整回答
反對 回復 2019-10-05
  • 3 回答
  • 0 關注
  • 932 瀏覽

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號