使用NSKeyedArchiver和Userdefault保存结构数组,其中对象符合NSCoding

Wol*_*ine 1 nsuserdefaults ios swift

作为面向协议的编程概念,我使用Struct创建了模型。

我想将“ ”的数组保存Struct到Userdefault中。但是我在此模型的数组的编码/解码中遇到问题。

这是我的模特Struct

struct Room {
    let name : String
    let id : String
    let booked : Bool
}
Run Code Online (Sandbox Code Playgroud)

我在这里创建了这样的扩展

extension Room {


func decode() -> Room? {
    let userClassObject = NSKeyedUnarchiver.unarchiveObject(withFile: RoomClass.path()) as? RoomClass
    return userClassObject?.room
}

func encode() {
    let personClassObject = RoomClass(room: self)
    NSKeyedArchiver.archiveRootObject(personClassObject, toFile: RoomClass.path())
}

class RoomClass: NSObject, NSCoding {

    var room : Room?

    init(room: Room) {
        self.room = room
        super.init()
    }

    class func path() -> String {
        let documentsPath = NSSearchPathForDirectoriesInDomains(FileManager.SearchPathDirectory.documentDirectory, FileManager.SearchPathDomainMask.userDomainMask, true).first
        let path = documentsPath?.appending(("/Room"))
        return path!
    }

    func encode(with aCoder: NSCoder) {
        aCoder.encode(room!.name, forKey: "name")
        aCoder.encode(room!.id, forKey: "Id")
        aCoder.encode(room!.booked, forKey: "booked")
    }

    required init?(coder aDecoder: NSCoder) {
        let _name = aDecoder.decodeObject(forKey: "name") as? String
        let _id = aDecoder.decodeObject(forKey: "Id") as? String
        let _booked = aDecoder.decodeBool(forKey: "booked")

        room = Room(name: _name!, id: _id!, booked: _booked)

        super.init()
    }
}
}
Run Code Online (Sandbox Code Playgroud)

当我尝试像这样保存arrRoomList(一个Room对象数组)时

        self.saveRooms(arrayRooms: arrRoomList)
Run Code Online (Sandbox Code Playgroud)

我得到这个错误

[_SwiftValue encodeWithCoder:]:无法识别的选择器已发送到实例

我也尝试过先对每个对象进行编码,然后尝试将它们保存为默认值,然后还会出现错误。

谁能指导我如何以正确的方式在Userdefaults中对Struct数组进行编码/解码,而不将其转换为Dictionary?

Cas*_*sey 6

您可以将结构设置为NSKeyedArchiver直接使用,如下所示:

struct Room {
    let name : String
    let id : String
    let booked : Bool
}

extension Room {
    func encode() -> Data {
        let data = NSMutableData()
        let archiver = NSKeyedArchiver(forWritingWith: data)
        archiver.encode(name, forKey: "name")
        archiver.encode(id, forKey: "id")
        archiver.encode(booked, forKey: "booked")
        archiver.finishEncoding()
        return data as Data
    }

    init?(data: Data) {
        let unarchiver = NSKeyedUnarchiver(forReadingWith: data)
        defer {
            unarchiver.finishDecoding()
        }
        guard let name = unarchiver.decodeObject(forKey: "name") as? String else { return nil }
        guard let id = unarchiver.decodeObject(forKey: "id") as? String else { return nil }
        booked = unarchiver.decodeBool(forKey: "booked")
        self.name = name
        self.id = id
    }
}
Run Code Online (Sandbox Code Playgroud)

要与UserDefaults一起使用,请像这样调用:

// to encode to data and save to user defaults
let room = Room(name: "asdf", id: "123", booked: true)
UserDefaults.standard.set(room.encode(), forKey: "room")

// to retrieve from user defaults
if let data = UserDefaults.standard.object(forKey: "room") as? Data {
    let room = Room(data: data)
}
Run Code Online (Sandbox Code Playgroud)

可以像这样保存/检索一系列房间:

func saveRooms(arrayRooms: [Room]) {
    let roomsData = arrayRooms.map { $0.encode() }
    UserDefaults.standard.set(roomsData, forKey: "rooms")
}

func getRooms() -> [Room]? {
    guard let roomsData = UserDefaults.standard.object(forKey: "rooms") as? [Data] else { return nil }
    return roomsData.flatMap { return Room(data: $0) }
}


// save 2 rooms to user defaults
let roomA = Room(name: "A", id: "123", booked: true)
let roomB = Room(name: "B", id: "asdf", booked: false)
saveRooms(arrayRooms: [roomA, roomB])

// get the rooms
print(getRooms())
Run Code Online (Sandbox Code Playgroud)