- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我在 Swift 中有一组对象,我正在使用 NSCoding 将其保存到磁盘,以便下次运行应用程序时可以将它们读入内存。正在保存的项目是一个 Volume 对象数组。 Volume 的每个实例都有一个已设置的 volumeNumber Int。 Volume 和 QA 类都继承自 NSObject 和 NSCoding。
如果应用程序在保存任何数据之前运行,它运行良好,当运行编码功能时,我的打印输出确认卷号似乎已正确保存(在我的示例中为 1 和 2)。但是,当我重新加载应用程序时,它会退出并显示错误: fatal error :在展开可选值时意外发现 nil 并突出显示:let volumeNumber = aDecoder.decodeObject(forKey: "volumeNumber") as!诠释
对我来说,应用程序似乎正在保存 Int,但由于某种原因,当应用程序尝试从内存中读取时无法找到它们。谁能看出我可能做错了什么?
这是我的音量类:
import Foundation
class Volume: NSObject, NSCoding {
let volumeNumber: Int
var completed: Bool
var questionsData: [QA]
init (volumeNumber: Int, completed: Bool, questionsData: [QA]) {
self.volumeNumber = volumeNumber
self.completed = completed
self.questionsData = questionsData
}
func percent() -> Int {
var correctAnswersTotal = 0
var percentage = 0
for question in questionsData {
if question.correctAnswer == question.selectedAnswer {
correctAnswersTotal += 1
}
}
percentage = ((correctAnswersTotal / questionsData.count) * 100) as Int
print("percentage is: \(percentage)")
return percentage
}
// MARK: NSCoding
public convenience required init?(coder aDecoder: NSCoder) {
print("Trying to read volumeNumber in decoder now")
let volumeNumber = aDecoder.decodeObject(forKey: "volumeNumber") as! Int
let completed = aDecoder.decodeObject(forKey: "completed") as! Bool
let questionsData = aDecoder.decodeObject(forKey: "questionsData") as! [QA]
self.init(volumeNumber: volumeNumber, completed: completed, questionsData: questionsData)
}
func encode(with aCoder: NSCoder) {
print("about to encode \(volumeNumber) in func encode")
aCoder.encode(volumeNumber, forKey: "volumeNumber")
aCoder.encode(completed, forKey: "completed")
aCoder.encode(questionsData, forKey: "questionsData")
}
}
这是我的 View Controller ,它试图保存/加载数据: 导入 UIKit
class VolumeTableViewController: UITableViewController {
static var volumesArray: [Volume] = [Volume(volumeNumber: 1, completed: false, questionsData: [QA(questionsText: "Question 1", answerText: ["Answer A", "Answer B", "Answer C", "Answer D"], correctAnswer: [true, false, false, false], selectedAnswer: [false, false, false, false])])]
override func viewDidLoad() {
print("view did load")
super.viewDidLoad()
if let loadedVolumes = VolumeTableViewController.read() {
print("Using loaded volumes...")
VolumeTableViewController.volumesArray = loadedVolumes
} else {
print("Setting up fresh set of volumes...")
VolumeTableViewController.volumesArray = [Volume(volumeNumber: 1, completed: false, questionsData: [QA(questionsText: "Vol1 Question 1", answerText: ["Answer A", "Answer B", "Answer C", "Answer D"], correctAnswer: [true, false, false, false], selectedAnswer: [false, false, false, false])]),
Volume(volumeNumber: 2, completed: false, questionsData: [QA(questionsText: "Vol 2 Question 1", answerText: ["Answer A", "Answer B", "Answer C", "Answer D"], correctAnswer: [true, false, false, false], selectedAnswer: [false, false, false, false])])]
}
// Uncomment the following line to preserve selection between presentations
// self.clearsSelectionOnViewWillAppear = false
// Uncomment the following line to display an Edit button in the navigation bar for this view controller.
// self.navigationItem.rightBarButtonItem = self.editButtonItem()
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
// MARK: - Table view data source
override func numberOfSections(in tableView: UITableView) -> Int {
return 1
}
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return VolumeTableViewController.volumesArray.count
}
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "VolumeCell", for: indexPath)
// Configure the cell...
if VolumeTableViewController.volumesArray[indexPath.row].completed {
let percent = VolumeTableViewController.volumesArray[indexPath.row].percent()
cell.textLabel?.text = "Volume \(indexPath.row + 1) \(percent) %"
}
cell.textLabel?.text = "Volume \(indexPath.row + 1)"
return cell
}
/*
// Override to support conditional editing of the table view.
override func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool {
// Return false if you do not want the specified item to be editable.
return true
}
// MARK: - Navigation
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
if segue.identifier == "goToQuestionsTableView" {
if let indexPath = self.tableView.indexPathForSelectedRow {
print("index selected for path for volume is: \(indexPath.row)")
let controller = segue.destination as! QuestionsTableViewController
controller.volume = indexPath.row
}
}
}
// Mark NSCODING
static func save() {
print("About to save...")
let DocumentsDirectory = FileManager().urls(for: .documentDirectory, in: .userDomainMask).first!
let ArchiveURL = DocumentsDirectory.appendingPathComponent("volumesData")
NSKeyedArchiver.archiveRootObject(VolumeTableViewController.volumesArray, toFile: ArchiveURL.path)
}
static func read() -> [Volume]? {
print("About to read()")
let DocumentsDirectory = FileManager().urls(for: .documentDirectory, in: .userDomainMask).first!
let ArchiveURL = DocumentsDirectory.appendingPathComponent("volumesData")
return NSKeyedUnarchiver.unarchiveObject(withFile: ArchiveURL.path) as? [Volume]
}
}
最佳答案
万一其他人遇到同样的问题,问题出在我如何编码我的数据上。特别是不将我的 Int 编码为使用 encodeCInt 等,请参阅解决此问题的代码更改:
// MARK: NSCoding
public convenience required init?(coder aDecoder: NSCoder) {
print("Trying to read volumeNumber in decoder now")
let volumeNumber = aDecoder.decodeInteger(forKey: "volumeNumber")
let completed = aDecoder.decodeBool(forKey: "completed")
let questionsData = aDecoder.decodeObject(forKey: "questionsData") as! [QA]
self.init(volumeNumber: volumeNumber, completed: completed, questionsData: questionsData)
}
func encode(with aCoder: NSCoder) {
print("about to encode \(volumeNumber) in func encode")
aCoder.encodeCInt(Int32(volumeNumber), forKey: "volumeNumber")
aCoder.encode(completed, forKey: "completed")
aCoder.encode(questionsData, forKey: "questionsData")
}
关于ios - 使用 NSCoding 将对象解码为 Int 时抛出的错误是 : fatal error: unexpectedly found nil while unwrapping an Optional value,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46393956/
在检查对象是否为 nil 时,有人使用 1: if (object == nil) { //... } 有人用 2: if (nil == object) { //... } 1和2有
func InsertApData(w http.ResponseWriter, r *http.Request) { decoder := json.NewDecoder(r.Body)
为什么 nil.to_s 返回 "",而 nil.inspect 返回 "nil"(当显然 .inspect 使用 .to_s 方法) 最佳答案 "inspect" method of the Obj
运行时间 https://play.golang.org/p/sl12vfS9vP package main import "fmt" func main() { err := run()
我可以将三元条件运算符用于 if {} else {} 语句,如下所示:a ? x : y,还是问题?回答 1 : 回答 2。 是否可以使用这种格式来检查,而不是 a 是 true 还是 false,
无法弄清楚这里出了什么问题。按照设计设置说明,用谷歌搜索我能想到的一切,仍然没有运气。 undefined method `email' for # Extracted source (around
今天写一些 rspec 时,我遇到了一些意外的行为,将日期(和时间)实例与 nil 进行比较。这是一个使用原始 ruby 的示例(没有 Rails 或其他库): user@MacBook-Work
我将数据类型的非零值分配给非可选属性,然后将其分配给可选属性,最后用所述数据实例化图像。当可选项通过 if-let 子句时,它的 block 执行,抛出错误: Fatal error: Unexpec
swift 5.1 。 考虑以下。 let x: Any? = nil let y: Any = x print("x \(x)") // x nil pri
请耐心听我解释这一点, 我正在创建一个聊天室,用户可以在其中上传照片供其他人查看。当用户点击图标时,他们可以将照片上传到我的 Firebase 数据库(这已成功完成,我已经对此进行了测试)。照片的 U
我的 xCode 5.0 目前遇到一个奇怪的问题:一个对象在控制台中似乎为 nil,但同时它可以通过代码访问。 图 1:对象似乎为零 图2:NSLog(@"%@", imgDownloader) 的输
我有一个实现协议(protocol)的类,以便添加 3 个变量。我专门设置了图像变量,调试器显示该变量存在,但是在我打印它时在代码中显示为 nil,我的 if let 语句也认为该变量为 nil。 @
这个问题在这里已经有了答案: Swift 2 ( executeFetchRequest ) : error handling (5 个答案) 关闭 7 年前。 开始在 SWIFT 中学习编码,每次
两者 (not 'nil) 和 (not nil) 求值为T,那么'nil和nil有什么区别吗?那么 ''nil 呢?如果 ''nil 的计算结果为 'nil,那么 ''nil 是否也应该计算为 ni
(if '(nil nil) 'print-true 'print-false) (if '(nil) 'print-true 'print-false) 在上面的代码
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: obj.nil? vs. obj == nil 我发现了一个问题 - == nil 和 nil 哪个更好?
此代码段抛出异常: x = nil jsoned = x.to_json puts 'x.to_json=' + jsoned.inspect puts 'back=' + JSON.parse(js
基本上我有一个对象想要传递给前端。我在后端记录了它,它不是空的,但是在前端,当我提醒它时,它变成了空。 ... presentation := &presentationStruct { Obje
我创建了一个自定义错误类型来包装错误,以便在 Golang 中更轻松地进行调试。当有错误要打印时它可以工作,但现在它会引起 panic 。 Demo type Error struct { E
写一个符合我当前问题的标题有点难..我有一个 main() 函数,它使用另一个包 (database_sql) 中的函数。该函数初始化一个全局变量 sql.DB*。初始化后,变量不为nil,但是对于其
我是一名优秀的程序员,十分优秀!