草庐IT

ios - 在 UI 交互中同时在后台同步时操作 Core Data Context 的最佳实践

coder 2024-01-28 原文

我已经在很多关于如何使用 NSManagedObjectContext 的文章和讨论中进行了研究。 ,但仍然无法为我的项目找到令人满意的架构。

在我的应用程序中,可以从三个来源修改数据,当同时发生冲突时按优先级排序(例如,云的优先级最低):

  1. 用户界面,
  2. BLE 消息,
  3. 来自云的 HTTP 响应

由于我仍然不是 iOS 开发专家,所以我尽量避免为每个源使用多个上下文。然而,经过数周的反复试验后,我很不情愿,但开始考虑是否真的需要采用多上下文方法。

一开始,我尝试使用context.perform { }在主上下文中执行所有数据更改操作(添加/更新/删除获取 除外)。我一直将获取作为同步功能,因为我希望数据获取是即时的,以便可以响应 UI。但是,在这种方法下,我偶尔会收到 "Collection <__NSCFSet: 0x000000000> was mutated while being enumerated"异常(我想这可能发生在 forEachmap 批处理数据函数中)。我还发现,当后台队列中有大量记录要更新时,这种方法仍然会阻塞 UI。

因此,我创建了一个后台上下文,并使用父子模型来操作数据。基本上主上下文(父)只负责获取数据,而后台上下文(子)通过backgroundContext.perform { }操作所有数据更改(添加/更新/删除) .这种方式解决了UI block的问题,但是collection mutated error还是时有发生,在这种结构下还会出现另外一个问题:比如我在ViewController A中添加一条数据记录,然后移动到View Controller B,app会崩溃,即使后台上下文没有完成添加数据记录,也立即获取相同的数据。

因此,我想就我的项目中的核心数据使用提出一些建议。在我的父子数据上下文模型下我做错了什么吗?或者,我是否应该不可避免地选择一个没有亲子关系的真正的多上下文模型?怎么做?

我的主上下文(父)和后台上下文(子)是这样启动的:

lazy var _context: NSManagedObjectContext = {
        return (UIApplication.shared.delegate as! AppDelegate).persistentContainer.viewContext
    }()

lazy var _backgroundContext: NSManagedObjectContext = {
    let ctx = NSManagedObjectContext(concurrencyType: NSManagedObjectContextConcurrencyType.privateQueueConcurrencyType)
    ctx.parent = self._context
    return ctx
}()

合并函数如下:

@objc func contextDidSaveContext(notification: NSNotification) {
    let sender = notification.object as! NSManagedObjectContext

    if sender === self._context {
        NSLog("******** Saved main Context in this thread")
        self._backgroundContext.perform {
            self._backgroundContext.mergeChanges(fromContextDidSave: notification as Notification)
        }
    } else if sender === self._backgroundContext {
        NSLog("******** Saved background Context in this thread")
        self._context.perform {
            self._context.mergeChanges(fromContextDidSave: notification as Notification)
        }
    }
    else {
        NSLog("******** Saved Context in other thread")
        self._context.perform {
            self._context.mergeChanges(fromContextDidSave: notification as Notification)
        }
        self._backgroundContext.perform {
            self._backgroundContext.mergeChanges(fromContextDidSave: notification as Notification)
        }
    }
}

核心数据结构的任何建议将不胜感激。

最佳答案

您探索过使用 NSFetchedResultsController 来获取数据吗? NSFetchedResultsController 提供了一种线程安全的方式来观察由于 CoreData 中的创建、更新或删除操作而导致的数据变化。 使用 NSFetchedResultsController 的委托(delegate)方法来更新 UI。 理想情况下,controllerDidChangeContent 委托(delegate)会给您一个更新 UI 的指示。 Documentation for reference

关于ios - 在 UI 交互中同时在后台同步时操作 Core Data Context 的最佳实践,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47027771/

有关ios - 在 UI 交互中同时在后台同步时操作 Core Data Context 的最佳实践的更多相关文章

  1. ruby-on-rails - 使用 Ruby on Rails 进行自动化测试 - 最佳实践 - 2

    很好奇,就使用ruby​​onrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提

  2. ruby - i18n Assets 管理/翻译 UI - 2

    我正在使用i18n从头开始​​构建一个多语言网络应用程序,虽然我自己可以处理一大堆yml文件,但我说的语言(非常)有限,最终我想寻求外部帮助帮助。我想知道这里是否有人在使用UI插件/gem(与django上的django-rosetta不同)来处理多个翻译器,其中一些翻译器不愿意或无法处理存储库中的100多个文件,处理语言数据。谢谢&问候,安德拉斯(如果您已经在ruby​​onrails-talk上遇到了这个问题,我们深表歉意) 最佳答案 有一个rails3branchofthetolkgem在github上。您可以通过在Gemfi

  3. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  4. ruby-on-rails - 如何在 ruby​​ 交互式 shell 中有多行? - 2

    这可能是个愚蠢的问题。但是,我是一个新手......你怎么能在交互式ruby​​shell中有多行代码?好像你只能有一条长线。按回车键运行代码。无论如何我可以在不运行代码的情况下跳到下一行吗?再次抱歉,如果这是一个愚蠢的问题。谢谢。 最佳答案 这是一个例子:2.1.2:053>a=1=>12.1.2:054>b=2=>22.1.2:055>a+b=>32.1.2:056>ifa>b#Thecode‘if..."startsthedefinitionoftheconditionalstatement.2.1.2:057?>puts"f

  5. ruby - 如何验证 IO.copy_stream 是否成功 - 2

    这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下

  6. Ruby 文件 IO 定界符? - 2

    我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的

  7. ruby-on-rails - 如何在 Ruby on Rails 中实现由 JSF 2.0 (Primefaces) 驱动的 UI 魔法 - 2

    按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭10年前。问题1)我想知道ruby​​onrails是否有功能类似于primefaces的gem。我问的原因是如果您使用primefaces(http://www.primefaces.org/showcase-labs/ui/home.jsf),开发人员无需担心javascript或jquery的东西。据我所知,JSF是一个规范,基于规范的各种可用实现,prim

  8. 叮咚买菜基于 Apache Doris 统一 OLAP 引擎的应用实践 - 2

    导读:随着叮咚买菜业务的发展,不同的业务场景对数据分析提出了不同的需求,他们希望引入一款实时OLAP数据库,构建一个灵活的多维实时查询和分析的平台,统一数据的接入和查询方案,解决各业务线对数据高效实时查询和精细化运营的需求。经过调研选型,最终引入ApacheDoris作为最终的OLAP分析引擎,Doris作为核心的OLAP引擎支持复杂地分析操作、提供多维的数据视图,在叮咚买菜数十个业务场景中广泛应用。作者|叮咚买菜资深数据工程师韩青叮咚买菜创立于2017年5月,是一家专注美好食物的创业公司。叮咚买菜专注吃的事业,为满足更多人“想吃什么”而努力,通过美好食材的供应、美好滋味的开发以及美食品牌的孵

  9. 使用canal同步MySQL数据到ES - 2

    文章目录一、概述简介原理模块二、配置Mysql使用版本环境要求1.操作系统2.mysql要求三、配置canal-server离线下载在线下载上传解压修改配置单机配置集群配置分库分表配置1.修改全局配置2.实例配置垂直分库水平分库3.修改group-instance.xml4.启动监听四、配置canal-adapter1修改启动配置2配置映射文件3启动ES数据同步查询所有订阅同步数据同步开关启动4.验证五、配置canal-admin一、概述简介canal是Alibaba旗下的一款开源项目,Java开发。基于数据库增量日志解析,提供增量数据订阅&消费。Git地址:https://github.co

  10. Get https://registry-1.docker.io/v2/: net/http: request canceled while waiting - 2

    1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里

随机推荐