如何在Swift 4中修复``无法识别的选择器发送到实例''

时间:2018-12-27 17:31:31

标签: swift xcode

我正在尝试使用UserDefaults创建一个简单的待办事项列表应用程序以存储任务。当我使用字符串数组进行存储时,我的程序正在运行,但是当我切换到存储自己的对象时,问题就开始了。该代码可能有一些冗余,但我仍在学习如何在设备上本地存储数据。

我只需要知道为什么会发生此错误,以及如何解决该错误。

我在代码中添加了我认为问题所在的位置,但是我将添加所有代码,以防其变得更加清晰。

SPOT 1:我根据控制台告诉我的内容在此处添加了objc(ABCItems)。

SPOT 2:我认为这里有一个问题,因为代码在这一点上失败了(我以后从未运行过的两个打印语句)。

import UIKit

class FirstViewController: UIViewController, UITableViewDelegate, UITableViewDataSource{

//SPOT 1

@objc(ABCItems)class Items: NSCoder{

    var title : String!
    var completed : Bool!

    required init(coder decoder: NSCoder) {
        self.title = decoder.decodeObject(forKey: "title") as? String
        self.completed = decoder.decodeObject(forKey: "completed") as? Bool
    }

    init(title: String, completed: Bool) {
        self.title = title
        self.completed = completed
    }

    func encodeWithCoder(coder: NSCoder) {
        if let title = title { coder.encode(title, forKey: "title") }
        if let completed = completed { coder.encode(completed, forKey: "completed") }
    }
}

let defaults = UserDefaults.standard;
var list = [Items]();
@IBOutlet weak var tableList: UITableView!

@IBAction func addItem(_ sender: Any) {
    let alert = UIAlertController(title: "Add an item", message: "", preferredStyle: .alert);
    alert.addTextField(configurationHandler: { (textField) in
        textField.placeholder = "Enter Task";
    })
    alert.addAction(UIAlertAction(title: "OK", style: .default, handler: { [weak alert] (_) in
        let textField = alert?.textFields![0];
        print("Text field: \(textField?.text ?? "")")
        if (textField?.text?.count)! > 0 {
            let item = Items(title: (textField?.text)!, completed: true);
            self.list.append(item);

            //SPOT 2

            let itemData = try! NSKeyedArchiver.archivedData(withRootObject: self.list, requiringSecureCoding: false);
            print("PASS 1");
            self.defaults.set(itemData, forKey: "list");
            print("PASS 2");
            self.tableList.reloadData();
        }
    }))
    self.present(alert, animated: true, completion: nil);
}

func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return list.count;
}

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = UITableViewCell(style: UITableViewCell.CellStyle.default, reuseIdentifier: "cell");
    cell.textLabel?.text = (list[indexPath.row] as AnyObject).title;
    cell.backgroundColor = UIColor (red: 0.18, green: 0.71, blue: 1.0, alpha: 1.0);
    return cell;
}

func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCell.EditingStyle, forRowAt indexPath: IndexPath) {
    if editingStyle == UITableViewCell.EditingStyle.delete {
        self.list.remove(at: indexPath.row);
        tableList.reloadData();
    }

}
override func viewDidLoad() {
    super.viewDidLoad()
    let itemDataRet = defaults.object(forKey: "list") as? NSData

    if let itemDataRet = itemDataRet {
        list = try! NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(itemDataRet as Data) as! [Items]
    }
}
}

这是我得到的输出: “-[[ABCItems encodeWithCoder:]:无法识别的选择器已发送到实例0x2829159e0”

1 个答案:

答案 0 :(得分:5)

采用NSCoding(不是NSCoder)的类必须是NSObject的子类。

class Items: NSObject, NSCoding { ...

但是在Swift 4中,强烈建议删除ObjC运行时并使用轻量级Codable协议将自定义结构或类序列化为Property ListJSON


该类可以简化为一个结构

struct Item : Codable {
    var title : String
    var completed : Bool
}

加载数据:

guard let data = UserDefaults.standard.data(forKey: "list") else {
    self.list = []
    return
}
do {
   self.list = try JSONDecoder().decode([Item].self, from: data)
} catch {
   print(error)
   self.list = []
}

保存数据:

do {
    let itemData = try JSONEncoder().encode(list)
    UserDefaults.standard.set(itemData, forKey: "list")
} catch {
    print(error)
}

一些注意事项:

  • 切勿将属性声明为在init方法中初始化的隐式未包装可选属性。使用常规的可选?或非可选。
  • 这是Swift:没有尾随的分号。
  • 结构和类应该以单数形式(Item)命名。