如何在swift中使用NSCoder编码枚举?

kfm*_*e04 63 enums persistence nscoder swift

背景

我正在尝试使用NSCoding协议对String样式的枚举进行编码,但是我遇到了从String转换回来的错误.

解码和编码时出现以下错误:

字符串不可转换为Stage

额外参数ForKey:在电话中

    enum Stage : String
    {
        case DisplayAll    = "Display All"
        case HideQuarter   = "Hide Quarter"
        case HideHalf      = "Hide Half"
        case HideTwoThirds = "Hide Two Thirds"
        case HideAll       = "Hide All"
    }

    class AppState : NSCoding, NSObject
    {
        var idx   = 0
        var stage = Stage.DisplayAll

        override init() {}

        required init(coder aDecoder: NSCoder) {
            self.idx   = aDecoder.decodeIntegerForKey( "idx"   )
            self.stage = aDecoder.decodeObjectForKey(  "stage" ) as String    // ERROR
        }

        func encodeWithCoder(aCoder: NSCoder) {
            aCoder.encodeInteger( self.idx,             forKey:"idx"   )
            aCoder.encodeObject(  self.stage as String, forKey:"stage" )  // ERROR
        }

    // ...

    }
Run Code Online (Sandbox Code Playgroud)

vac*_*ama 67

您需要将枚举转换为原始值和从原始值转换.在Swift 1.2(Xcode 6.3)中,这看起来像这样:

class AppState : NSObject, NSCoding
{
    var idx   = 0
    var stage = Stage.DisplayAll

    override init() {}

    required init(coder aDecoder: NSCoder) {
        self.idx   = aDecoder.decodeIntegerForKey( "idx" )
        self.stage = Stage(rawValue: (aDecoder.decodeObjectForKey( "stage" ) as! String)) ?? .DisplayAll
    }

    func encodeWithCoder(aCoder: NSCoder) {
        aCoder.encodeInteger( self.idx, forKey:"idx" )
        aCoder.encodeObject(  self.stage.rawValue, forKey:"stage" )
    }

    // ...

}
Run Code Online (Sandbox Code Playgroud)

Swift 1.1(Xcode 6.1),as代替as!:

    self.stage = Stage(rawValue: (aDecoder.decodeObjectForKey( "stage" ) as String)) ?? .DisplayAll
Run Code Online (Sandbox Code Playgroud)

Swift 1.0(Xcode 6.0)使用toRaw()fromRaw()像这样:

    self.stage = Stage.fromRaw(aDecoder.decodeObjectForKey( "stage" ) as String) ?? .DisplayAll

    aCoder.encodeObject( self.stage.toRaw(), forKey:"stage" )
Run Code Online (Sandbox Code Playgroud)

  • `as!`正在施放`String`,它确实存在,所以演员表会成功.`Stage(rawValue:"someString")`返回一个可选项,因为该字符串可能没有定义有效的枚举值.你必须打开那个可选项.*nil coalescing运算符*将该可选项替换为unwrapped版本(如果存在)或".DisplayAll"(如果不存在). (2认同)

Adr*_*ers 9

更新Xcode 6.3,Swift 1.2:

self.stage = Stage(rawValue: aDecoder.decodeObjectForKey("stage") as! String) ?? .DisplayAll
Run Code Online (Sandbox Code Playgroud)

请注意 as!


小智 7

这是Swift 4.2的解决方案.如其他答案中所述,问题是您尝试直接为stage变量分配解码字符串,并尝试将stage变量强制转换为encodeWithCoder方法中的字符串.您需要使用原始值.

enum Stage: String {
    case DisplayAll = "Display All"
    case HideQuarter = "Hide Quarter"
    case HideHalf = "Hide Half"
    case HideTwoThirds = "Hide Two Thirds"
    case HideAll = "Hide All"
}

class AppState: NSCoding, NSObject {
    var idx = 0
    var stage = Stage.DisplayAll

    override init() {}

    required init(coder aDecoder: NSCoder) {
        self.idx = aDecoder.decodeInteger(forKey: "idx")
        self.stage = Stage(rawValue: aDecoder.decodeObject(forKey: "stage") as String)
    }

    func encodeWithCoder(aCoder: NSCoder) {
        aCoder.encode(self.idx, forKey:"idx")
        aCoder.encode(self.stage.rawValue, forKey:"stage")
    }

    // ...

}
Run Code Online (Sandbox Code Playgroud)

  • 如果枚举没有行值怎么办?在我的应用程序中,它只是一个枚举:“enum SomeType { case1, case2, case3 }”,像这样?这种情况该如何处理呢? (3认同)