问题
*** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: '[ valueForUndefinedKey:]: this class is not key value coding-compliant for the key active.'
感谢@GrahamPerks answer对此SO question ,我在我的代码中插入了一个异常断点,现在在这一行暂停执行...
static var entityActive: Bool {
return entity.value(forKey: "active") as! Bool // <-- PAUSES AT THIS LINE
}
这显然需要进一步解释......
背景
我正在编写一个使用通用 TableView 数据源和委托(delegate)的 Core Data 应用程序,或多或少是根据 Florian Kugler 于 2017 年由 objc.io 出版的书“Core Data”设置的.
我已经成功地将三个单独的 UITableViewController 链接到这个通用数据源/委托(delegate)。我使用的是一个主 Storyboard,这三个 Controller 链接到三个 UISplitViewController 主视图/详细 View 。
我已经删除了 Storyboard UITableView 中自动生成的数据源和委托(delegate)连接(尽管我是否保留或删除这些连接似乎没有什么区别)。
如果我注释掉上面的 static var entityActive 代码,项目将成功构建并运行。
我的代码使用 UITableViewDelegate 方法 tableView(_, willDisplay:, forRowAt:) 来改变文本的 .textColor 和 .backgroundColor 单元格,基于属性“active”,存储为每个实体的 bool 标量类型值。
为清楚起见,我尝试为实体的每个 NSManagedObject 获取“active”属性(我的数据模型中的每个实体都有一个公共(public)属性“active”)。然后使用每个实体的“事件”属性(Bool true 或 false)的值来格式化 if.. .else 语句在下面的代码中。
为了尝试最大化代码重用和最小化代码重复,我还将此委托(delegate)方法放在我的通用数据源/委托(delegate)类中。
func tableView(_ tableView: UITableView, willDisplay cell: UITableViewCell, forRowAt indexPath: IndexPath) {
let entityObjectActive = T.entityActive
if entityObjectActive == true {
cell.textLabel?.textColor = UIColor.black
cell.detailTextLabel?.textColor = UIColor.darkGray
cell.backgroundColor = UIColor.white
} else if entityObjectActive == false {
cell.textLabel?.textColor = UIColor.lightGray
cell.detailTextLabel?.textColor = UIColor.lightGray
cell.backgroundColor = UIColor.clear
}
}
编译器不会提示。
我似乎能够使用通用类型 T(代表核心数据实体)将静态属性 entityActive 与我的 entityObjectActive - 所以这似乎有效...
let entityObjectActive = T.entityActive
为了确认,我有以下内容:
通用数据源类....
class MyDataSource<T: Managed,
Delegate: TableViewDataSourceDelegate>:
NSObject,
UITableViewDataSource,
UITableViewDelegate,
NSFetchedResultsControllerDelegate {
// lots of code...
}
协议(protocol)...
protocol Managed: class, NSFetchRequestResult {
static var entity: NSEntityDescription { get }
static var entityActive: Bool { get } }
}
和扩展...
extension Managed where Self: NSManagedObject {
static var entity: NSEntityDescription { return entity() }
static var entityActive: Bool {
return entity.value(forKey: "active") as! Bool
}
}
尝试解决问题
我读了一些书试图解决我的问题,包括对许多博客的回顾(关于如何设置通用泛型和通用 TableView 数据源)和大量 SO 问答,特别是...
How to fix Error: this class is not key value coding-compliant for the key tableView.'
Uncaught exception: This class is not key value coding-compliant
setValue:forUndefinedKey: this class is not key value coding-compliant for the key
似乎所有 SO Q&A 都与 Storyboard中的 IB 连接问题直接相关。也许这也是我的问题 - 但如果是这样的话,在我看来它隐藏得很好。
有什么帮助或建议吗?
PS:我在长期使用 Obj-C 编写代码后正在学习 Swift,我真的在努力应对泛型协议(protocol)扩展范式转变,所以我对泛型类型的使用可能不正确?
最佳答案
感谢那些将我引向这个解决方案的评论...
我的修订协议(protocol) Managed...
protocol Managed: class, NSFetchRequestResult {
static var entity: NSEntityDescription { get }
var attributeActive: Bool { get } // <-- REMOVED static
}
我对 Managed 的修订扩展...
更新 - 添加 var attributeActive 到 Managed 扩展...
extension Managed where Self: NSManagedObject {
static var entity: NSEntityDescription { return entity() }
var attributeActive: Bool {
guard let attribute = self.value(forKey: "active") as? Bool else {
return false // in case key "active" is not set
}
return attribute
}
}
更新 - 从不再需要的托管对象扩展中删除 var attributeActive...
我的三个核心数据实体中的每一个的修订扩展(其数据显示在三个单独的 UITableViewController 中的每一个中)...
extension <<DataModelEntity>>: Managed {
public var attributeActive: Bool {
return self.active
}
@NSManaged public var active: Bool
@NSManaged public var <<OTHER DATA MODEL ENTITY ATTRIBUTES>> //...
// ...etc.
}
为了清楚起见,这里可能值得注意的是,数据模型中每个实体的 Codegen 值都设置为 Manual/None,所以我手动 1 为每个实体准备类和扩展。
1 当我手动写的时候,我的意思是我使用了Editor下的Create Managed Object Subclass...功能 Xcode 中的菜单,然后手动输入我的 Managed 协议(protocol) stub 。
最后,我的 REVISED 通用数据源委托(delegate)类...
class MyDataSource<T: Managed,
Delegate: TableViewDataSourceDelegate>:
NSObject,
UITableViewDataSource,
UITableViewDelegate,
NSFetchedResultsControllerDelegate {
// lots of code...
func tableView(_ tableView: UITableView, willDisplay cell: UITableViewCell, forRowAt indexPath: IndexPath) {
let object = fetchedResultsController.object(at: indexPath)
let objectActive = object.attributeActive
if objectActive == true {
cell.textLabel?.textColor = UIColor.black
cell.detailTextLabel?.textColor = UIColor.darkGray
cell.backgroundColor = UIColor.white
} else if entityObjectActive == false {
cell.textLabel?.textColor = UIColor.lightGray
cell.detailTextLabel?.textColor = UIColor.lightGray
cell.backgroundColor = UIColor.clear
}
}
// lots more code...
}
如果有人仍在阅读这篇文章,也许您对这个解决方案的原因感兴趣?
坦率地说,我自己仍在弄清楚细节,并计划在未来添加更简洁/准确的原因,因为我对 swift 泛型、协议(protocol)和扩展的理解有所提高,但现在我提供以下内容......
正如评论中所指出的,我错误地尝试根据实体描述中的数据模型属性获取实体属性。这就像通过向乐高盒子询问其中一 block 积木的特性来尝试获得乐高积木的颜色或大小。 “哪 block 砖?”盒子可能会问,如果乐高盒子可以问这样的事情。
基本上我犯了几个错误。我不明白静态定义对我的变量的影响,我也没有正确理解我的 Managed 协议(protocol)和扩展如何与采用该协议(protocol)的任何类交互。
关于ios - Swift 通用 TableView 数据源和委托(delegate),协议(protocol)支持 NSManagedObjects,无法获取实体属性,运行时错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53271766/
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-
大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje
在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/
我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r
我正在尝试在我的centos服务器上安装therubyracer,但遇到了麻烦。$geminstalltherubyracerBuildingnativeextensions.Thiscouldtakeawhile...ERROR:Errorinstallingtherubyracer:ERROR:Failedtobuildgemnativeextension./usr/local/rvm/rubies/ruby-1.9.3-p125/bin/rubyextconf.rbcheckingformain()in-lpthread...yescheckingforv8.h...no***e
我花了三天的时间用头撞墙,试图弄清楚为什么简单的“rake”不能通过我的规范文件。如果您遇到这种情况:任何文件夹路径中都不要有空格!。严重地。事实上,从现在开始,您命名的任何内容都没有空格。这是我的控制台输出:(在/Users/*****/Desktop/LearningRuby/learn_ruby)$rake/Users/*******/Desktop/LearningRuby/learn_ruby/00_hello/hello_spec.rb:116:in`require':cannotloadsuchfile--hello(LoadError) 最佳
Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/