草庐IT

ios - 从CloudKit删除记录时如何防止孤儿?

coder 2024-01-22 原文

CloudKit WWDC视频建议像这样实现同步:

  • 跟踪本地更改
  • 将更改发送到服务器
  • 解决冲突
  • 使用CKFetchRecordChangesOperation来获取服务器更改
  • 应用服务器更改
  • 保存服务器更改 token

  • 我在我的应用程序中遵循了这种模式,但是我遇到了删除和父子关系的问题。

    假设我们有一个分为几类的书籍清单。每本书必须完全属于一个类别。

    我从这样的数据开始:
    SERVER
    Thrillers: "Look Out!", "Secret Spy"
    Non-Fiction: "Sailing the Seas", "Gardening Adventures"
    Computer Programming: <empty>
    

    如您所见,最终类别为空。假设我有两个具有此数据精确副本的设备。

    现在,在设备1上,用户将一本书CloudKit Sync添加到“计算机编程”中:
    DEVICE 1
    Thrillers: "Look Out!", "Secret Spy"
    Non-Fiction: "Sailing the Seas", "Gardening Adventures"
    Computer Programming: "CloudKit Sync"
    

    但是在设备2上,用户完全删除了“计算机编程”类别(该类别为空,因此从设备2的角度来看是可以的):
    DEVICE 2
    Thrillers: "Look Out!", "Secret Spy"
    Non-Fiction: "Sailing the Seas", "Gardening Adventures"
    

    设备1首先进行同步,因此它将创建一个新的Book条目,其parent字段设置为Computer Programming

    但是现在设备2开始其同步过程。它将更改应用于服务器,因此删除了与“计算机编程”相对应的CKRecord。这与设备2的世界观一致,在世界观中类别为空,可以删除。

    但是,当它从服务器删除此类别时,就设备1的世界观和服务器本身而言,这没有任何意义。现在有一本名为CloudKit Sync的孤儿书,它有指向父级的悬挂指针。

    如果我遵循WWDC的Apple建议,如何避免这种情况?根据同步的顺序,我可以很容易地得到一本孤立的书和一个无效的父母参考的不一致状态。

    我想要发生的是从设备2发出的Delete命令返回一个错误,告诉我我要孤立一本书并完全阻止该操作发生,因此我可以采取一些措施来解决这种情况。

    那可能吗?还有另一种方法可以解决这个问题吗?

    最佳答案

    是的,您想要的设备2行为是可能的。我看到了在您的方案中将要使用的cloudkit的三个方面。让我们先看看这些,然后再看它们在您的场景中的用法。

    首先,假设两个(或所有)设备都已订阅对适当记录的更改,则将通知每个设备有人添加或删除了某些内容。然后,接收到警报的设备将有机会决定如何处理它。 (从本地视图中删除它,在服务器上替换它,等等)

    其次,您可以使用savePolicy上的CKModifyRecordOperation设置处理冲突的行为。您可以指定最后的更改是否应该覆盖较旧的记录,引发错误等。有关这三个选项,请参见https://developer.apple.com/documentation/cloudkit/ckrecordsavepolicy?language=objc。 (我仅在两个用户修改一条 public 记录的上下文中使用过此方法,但是在另一位用户更新该记录后删除该文件将引发server record changed错误)。

    第三,假设您已经配置了上述的savePolicy,那么服务器更改 token 本身就是。我发现最容易将更改 token 设想为最后修改的时间戳。 “这种记录的我的副本最近一次修改是在晚上10:42”。根据您在上述savePolicy中选择的覆盖选项,设备将收到NSError Server Record Changed,以警告您服务器上的版本来自晚上10:56,并且您的本地版本可能不再有效。

    产生的NSError中的userInfo包含有问题的记录的3个版本:服务器上的当前版本,您尝试提交的版本以及 public 祖先版本。苹果公司的指南说,由开发人员决定如何合并这些信息。但从理论上讲,您将能够区分更改,确定要保留的内容,然后提交新操作。

    关于您的特定情况:假设您完全授权并信任dev1和dev2删除记录,那么我将订阅创建和删除事件,并设置savePolicy在尝试发生冲突的更改时引发错误。在这种情况下,设备1将添加记录,而设备2将接收新记录的通知。如果设备2只是尝试删除旧记录,则它应失败并显示server record changed错误,您可以向用户显示为

    “其他人修改了此记录,您真的要删除它吗?
    (是/否)。”

    在继续操作之前,设备2必须刷新记录(并接收新的记录更改 token )。此后,如果设备2仍要删除新记录,可以删除,但随后将通过上述订阅将更改通知给设备1。然后,设备1将新记录下载到其本地视图中(或在这种情况下,从中删除旧记录)。订阅通知可能会警告用户1:

    “您的记录Foo被Bar删除了”

    即使事件实际上是同时发生的,这也将起作用,因为其中一个更改将首先应用于服务器,而另一设备的 token 将立即过时。因此,如果设备2首先设法删除了记录,则设备1修改记录的尝试将失败,并带有server record changed,因为设备1的更改 token 现在已过期。设备1的错误处理程序将必须根据您的业务规则决定是接受删除还是继续创建新记录。也许向用户1询问类似以下内容:

    从服务器上删除了“计算机编程”。您想重新创建吗
    它?

    此时,user1可以发送紧急电子邮件,要求其他用户停止删除其新创建的记录,而user2可以要求人们停止重新创建刚刚“清理”的记录。 :)

    您可能会变得更加复杂,可能会给设备1优先于设备2的优先级,这样,当通知设备1记录已删除时,设备1会将记录重新写入服务器。如果您有多个具有删除权限的用户,则可以确定优先顺序并构建适当的错误/通知处理程序。但是,这似乎非常复杂且容易出错。可能会发生自动响应的循环(创建,删除,创建,删除,创建,删除)。我仅将其作为假设示例,而不作为建议!

    最后,作为另一个示例,我的应用程序具有不同的方案。我的记录是游戏 session 。所有播放器都需要对 session 数据进行读取访问,但是只有始发者才可以选择完全删除记录。因此,您可能会考虑是否确实授权多个用户删除共享记录。

    关于ios - 从CloudKit删除记录时如何防止孤儿?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48817882/

    有关ios - 从CloudKit删除记录时如何防止孤儿?的更多相关文章

    1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

      我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

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

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

    3. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

      关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

    4. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

      给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

    5. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

      我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

    6. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

      我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

    7. ruby - 如何指定 Rack 处理程序 - 2

      Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

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

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

    9. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

      我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

    10. ruby - 我可以使用 Ruby 从 CSV 中删除列吗? - 2

      查看Ruby的CSV库的文档,我非常确定这是可能且简单的。我只需要使用Ruby删除CSV文件的前三列,但我没有成功运行它。 最佳答案 csv_table=CSV.read(file_path_in,:headers=>true)csv_table.delete("header_name")csv_table.to_csv#=>ThenewCSVinstringformat检查CSV::Table文档:http://ruby-doc.org/stdlib-1.9.2/libdoc/csv/rdoc/CSV/Table.html

    随机推荐