我正在尝试使用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”
答案 0 :(得分:5)
采用NSCoding
(不是NSCoder
)的类必须是NSObject
的子类。
class Items: NSObject, NSCoding { ...
但是在Swift 4中,强烈建议删除ObjC运行时并使用轻量级Codable
协议将自定义结构或类序列化为Property List
或JSON
。
该类可以简化为一个结构
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
方法中初始化的隐式未包装可选属性。使用常规的可选?
或非可选。Item
)命名。