jjzjj

ios - Swift 通用 TableView 数据源和委托(delegate),协议(protocol)支持 NSManagedObjects,无法获取实体属性,运行时错误

coder 2024-01-28 原文

问题

*** 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 truefalse)的值来格式化 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 attributeActiveManaged 扩展...

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/

有关ios - Swift 通用 TableView 数据源和委托(delegate),协议(protocol)支持 NSManagedObjects,无法获取实体属性,运行时错误的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

    我在从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""-

  3. ruby-on-rails - Rails 常用字符串(用于通知和错误信息等) - 2

    大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje

  4. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

  5. ruby-on-rails - 无法使用 Rails 3.2 创建插件? - 2

    我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby​​1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在

  6. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  7. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行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

  8. ruby-on-rails - 无法在centos上安装therubyracer(V8和GCC出错) - 2

    我正在尝试在我的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

  9. ruby - 无法让 RSpec 工作—— 'require' : cannot load such file - 2

    我花了三天的时间用头撞墙,试图弄清楚为什么简单的“rake”不能通过我的规范文件。如果您遇到这种情况:任何文件夹路径中都不要有空格!。严重地。事实上,从现在开始,您命名的任何内容都没有空格。这是我的控制台输出:(在/Users/*****/Desktop/LearningRuby/learn_ruby)$rake/Users/*******/Desktop/LearningRuby/learn_ruby/00_hello/hello_spec.rb:116:in`require':cannotloadsuchfile--hello(LoadError) 最佳

  10. ruby - Sinatra:运行 rspec 测试时记录噪音 - 2

    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/

随机推荐