如何让 NSBatchDeleteRequest 删除并允许您再次添加回对象

How to get a NSBatchDeleteRequest to delete and allow you to add back to an object again

我有以下代码。

let workorderFetchRequest   = NSFetchRequest<NSFetchRequestResult>(entityName: "Workorders")
                let deleteWorkOrderRequest  = NSBatchDeleteRequest(fetchRequest: workorderFetchRequest) //Deletes ALL workorders

                //Question is will it delete all references to the work orders correctly now?
                try context.execute(deleteWorkOrderRequest)

                //Save the work order objects (overwriting any old ones with same id if needed)
                do {
                    try context.save()
                    print("Saved context with workorders")
                } catch let error as NSError  {
                    print("Could not save \(error), \(error.userInfo)")
                } catch {
                    print("Could not save, unknown error")
                }

                print("deleted references")

工单看起来像这样...

import Foundation
import CoreData

extension Workorders {

    @nonobjc public class func fetchRequest() -> NSFetchRequest<Workorders> {
        return NSFetchRequest<Workorders>(entityName: "Workorders");
    }

    @NSManaged public var address: String?
    @NSManaged public var client_id: Int64
    @NSManaged public var desc: String?
    @NSManaged public var id: Int64
    @NSManaged public var phone: String?
    @NSManaged public var signaturepath: String?
    @NSManaged public var lat: String?
    @NSManaged public var lng: String?
    @NSManaged public var service: Service?
    @NSManaged public var pictures: NSSet?
    @NSManaged public var videos: NSSet?

}

// MARK: Generated accessors for pictures
extension Workorders {

    @objc(addPicturesObject:)
    @NSManaged public func addToPictures(_ value: Pictures)

    @objc(removePicturesObject:)
    @NSManaged public func removeFromPictures(_ value: Pictures)

    @objc(addPictures:)
    @NSManaged public func addToPictures(_ values: NSSet)

    @objc(removePictures:)
    @NSManaged public func removeFromPictures(_ values: NSSet)

}

// MARK: Generated accessors for videos
extension Workorders {

    @objc(addVideosObject:)
    @NSManaged public func addToVideos(_ value: Videos)

    @objc(removeVideosObject:)
    @NSManaged public func removeFromVideos(_ value: Videos)

    @objc(addVideos:)
    @NSManaged public func addToVideos(_ values: NSSet)

    @objc(removeVideos:)
    @NSManaged public func removeFromVideos(_ values: NSSet)

}

我让它引用了一个看起来像这样的服务对象...

import Foundation
import CoreData

extension Service {

    @nonobjc public class func fetchRequest() -> NSFetchRequest<Service> {
        return NSFetchRequest<Service>(entityName: "Service");
    }

    @NSManaged public var id: Int64
    @NSManaged public var name: String?
    @NSManaged public var templatedata: String?
    @NSManaged public var workorder: Workorders?

}

当我运行我在工作单模型上的删除代码设置为无效作为删除规则 ...(参考下图)

当我 运行 批量删除时,它似乎没有正确取消服务。当我尝试在工单上设置新服务时,它崩溃了...

libc++abi.dylib:以 NSException

类型的未捕获异常终止

我确定是这种情况,因为如果我将删除规则设置为级联工作单,那么问题就解决了。但是,我想保留旧服务只是使引用无效。

为什么我不能在工作订单上设置新服务而不会崩溃?

下面附上的是通过查看控制台消息崩溃的代码区域,我已将其缩小到尝试设置服务的那一刻。

另请注意,我正在使用 NSMergeByPropertyObjectTrumpMergePolicy 因为我打算再次为工单设置相同的 ID 服务,所以当我创建一个具有相同 ID 的新服务时如果我的理解是正确的,应该只是覆盖旧服务的数据。这可能是问题所在吗?如果是,我该如何解决?

好吧,我意外地找到了问题的答案。

看来您不必为一个对象提前保存上下文以供另一个对象引用,事实上...您不应该这样做!

这样做显然会使 nsbatchdelete 不同步。我不确定如何解决这个问题,并且希望得到比我自己更好的答案。

总之我摆脱了早...

do {
                    try context.save()
                    print("Saved context with workorders")
                } catch let error as NSError  {
                    print("Could not save \(error), \(error.userInfo)")
                } catch {
                    print("Could not save, unknown error")
                }

在我保存服务之后,因为它显然是不必要的而且有害。

这确实引出了一个问题,如果我在此之后执行另一个事务,我需要保存上下文并删除,这个错误会再次发生吗?

我从你的尝试中发现了一些问题。

起初,我认为您需要修改实体名称。它们应该是单一类型。

例如,您为他们使用了 "Workorders" 或 "Pictures"。 应该是 "Workorder" 和 "Picture".

主要问题是您在设置正确关系之前同步了托管对象。

例如,您创建了一个服务对象,它的工单关系还为零。 在这一点上,您保存了上下文。 如果你有可选的关系,它可能会发生。

并且您创建了工作单对象并将现有服务绑定到它的关系。

Core Data 尝试将新工作单绑定到服务对象的工作单关系,反之亦然。

所以它应该在一次交易中完成。 最后就是关系建立完成后需要保存一次context。

因此删除服务的 saveContext 行。

假设你像现在一样做了。 使用新服务保存上下文后,您创建了 workorder1 并为其关系设置了服务。然后应用程序崩溃或被某些东西停止了。 在这些情况下会导致数据完整性问题。

希望本文能帮助您理解并解决此问题。

干杯!