gpt4 book ai didi

swift - 如何在 Swift 4 中修复 'unrecognized selector sent to instance'

转载 作者:行者123 更新时间:2023-11-28 09:48:47 36 4
gpt4 key购买 nike

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

我只需要知道为什么会出现此错误以及我应该如何修复它。

我已经在我的代码中添加了我认为问题所在的点,但我将添加我的所有代码以防它使它更清晰。

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

点 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”

最佳答案

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

class Items: NSObject, NSCoding { ...

但在 Swift 4 中,强烈建议放弃 ObjC 运行时并使用轻量级 Codable 协议(protocol)将自定义结构或类序列化为 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)。

关于swift - 如何在 Swift 4 中修复 'unrecognized selector sent to instance',我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53948770/

36 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com