- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
从10.7上带有CoreData模板的标准Xcode基于文档的应用程序开始,我遇到了一些令人沮丧的行为。我敢肯定,这是我所忽略的简单事物。
假设在我的NSPersistentDocument子类中,我有类似这样的内容,它连接到窗口中的一个按钮:
- (IBAction)doStuff:(id)sender
{
NSEntityDescription* ent = [[self.managedObjectModel entitiesByName] valueForKey: @"MyEntity"];
NSManagedObject* foo = [[[NSManagedObject alloc] initWithEntity: ent insertIntoManagedObjectContext: self.managedObjectContext] autorelease];
[self.managedObjectContext save: NULL];
}
This NSPersistentStoreCoordinator has no persistent stores. It cannot perform a save operation.
我得到了。我们尚未保存,没有持久性存储。说得通。
- (IBAction)doStuff:(id)sender
{
NSEntityDescription* ent = [[self.managedObjectModel entitiesByName] valueForKey: @"MyEntity"];
NSManagedObject* foo = [[[NSManagedObject alloc] initWithEntity: ent insertIntoManagedObjectContext: self.managedObjectContext] autorelease];
}
- (IBAction)doOtherStuff:(id)sender
{
[self.managedObjectContext save: NULL];
}
最佳答案
所以我无所事事,包括尝试@Aderstedt的建议。这种方法没有用,因为伪造通知似乎只是告诉接收上下文“嘿,检查持久性存储,我已经更新了它们!”,实际上却没有,因为没有。我最终找到了可行的方法。不幸的是,它依赖于仅Lion的功能,因此我仍在寻找一种不需要Lion的方法。
背景
我想使用NSPersistentDocument方法。尽管我在任何地方都没有找到明确记录的文档,但我发现了一些论坛帖子,并获得了大量的经验证据,证明您不能在属于NSPersistentDocument的上下文中调用-[NSManagedObjectContext save:]
。正如问题中提到的,如果您在保存文档之前调用它,它将没有任何存储,因此保存将失败。即使在存储存在之后,也可以通过直接保存上下文(而不是通过文档保存API)来有效地更改NSPersistentDocument背后的磁盘上表示形式,并且您会得到如下文件弹出窗口:
文件已被另一个应用程序修改
简而言之,NSPersistentDocument期望控制关联的NSManagedObjectContext本身的保存动作。
同样值得一提的是:这里的目标是确保UI使用的上下文不会触发(或至少最少)I / O,以保持响应速度。我最终选择的模式是具有3个上下文。 NSPersistentDocument拥有的一个上下文,该上下文将负责与文档一起执行文件I / O。用于绑定UI的第二个有效只读上下文。 (我认识到很多人都希望UI能够改变模型,所以这对他们来说可能不那么令人兴奋,但这对我来说不是必需的。)第三个上下文用于后台线程,该线程从Web异步加载数据服务,并希望将其推送到其他上下文中,以便既可以将其保存在磁盘上也可以在UI中显示,而不会潜在地阻止网络I / O上的UI。
仅狮子解决方案
Lion的CoreData实现中新的父/子NSManagedObjectContext功能非常适合此功能。我用并发类型为NSPrivateQueueConcurrencyType的新MOC替换了NSPersistentDocument的NSManagedObjectContext。这将是“根”上下文。然后,使用NSMainQueueConcurrencyType并发性创建UI上下文,并将其作为根上下文的子级。最后,我将网络加载上下文作为NSPrivateQueueConcurrencyType上下文,它是UI上下文的子级。它的工作方式是我们在后台启动网络加载操作,它会更新网络上下文。完成后,将保存上下文。对于父/子关系,保存子上下文会将更改推入父上下文(UI上下文)中,但不会将父上下文保存到存储中。就我而言,我还从网络上下文中监听NSManagedObjectContextDidSaveNotification通知,然后告诉其父级也要保存(这会将UI上下文中的更改推送到根/磁盘上下文中,但不会将其保存到磁盘中。)
在这一系列事件的结尾,所有上下文都是一致的,我们仍然没有真正保存底层的根上下文,因此我们没有违反NSPersistentDocument的管理磁盘上的角色表示。
一个陷阱是,如果您想防止子上下文的保存生成撤消操作(即这是一个网络加载操作,则没有要撤消的操作),则必须在沿链向上传播更改时在每个父上下文上禁用UndoRegistration。
狮子前的努力
我真的很想为这个问题找到一个兼容Lion的解决方案。在放弃之前,我尝试了一些尝试。我首先尝试将内存存储与文档初始化上的PSC关联,以便可以在保存文档之前执行NSManagedObjectContext保存,然后在第一次保存时迁移内存中的存储。这部分效果很好。但是一旦存在磁盘上的存储,这种方法就是伪造的,因为在将其保存到磁盘后,我们会遇到同样的问题,即与文档连接的NSPersistentDocument拥有的PSC的MOC的任何保存都必须由文档完成。
我还尝试使用NSManagedObjectContextObjectsDidChangeNotification有效负载破解一种将更改从一个上下文移动到另一个上下文的机制。尽管我能够使它起作用(对于“工作”的一些名义上的定义),但是我看到这种方法迫在眉睫。具体来说,一次迁移这些更改很容易,但是如果在保存操作之前再次更改了该怎么办?然后,我将坚持维护源上下文中OID到目标上下文中OID的长期映射。这真的很难看。如果有人感兴趣,这是我想出的:
@interface NSManagedObjectContext (MergeChangesFromObjectsDidChangeNotification)
- (void)mergeChangesFromObjectsDidChangeNotification: (NSNotification*)notification;
@end
@implementation NSManagedObjectContext (MergeChangesFromObjectsDidChangeNotification)
- (void)mergeChangesFromObjectsDidChangeNotification: (NSNotification*)notification
{
if (![NSManagedObjectContextObjectsDidChangeNotification isEqual: notification.name])
return;
if (notification.object == self)
return;
NSManagedObjectContext* sourceContext = (NSManagedObjectContext*)notification.object;
NSAssert(self.persistentStoreCoordinator == sourceContext.persistentStoreCoordinator, @"Can't merge changes between MOCs with different persistent store coordinators.");
[sourceContext lock];
// Create object in the local context to correspond to inserted objects...
NSMutableDictionary* foreignOIDsToLocalOIDs = [NSMutableDictionary dictionary];
for (NSManagedObject* foreignMO in [[notification userInfo] objectForKey: NSInsertedObjectsKey])
{
NSManagedObjectID* foreignOID = foreignMO.objectID;
NSManagedObject* localMO = [[[NSManagedObject alloc] initWithEntity: foreignMO.entity insertIntoManagedObjectContext: self] autorelease];
[foreignOIDsToLocalOIDs setObject: localMO.objectID forKey: foreignOID];
}
// Bring over all the attributes and relationships...
NSMutableSet* insertedOrUpdated = [NSMutableSet set];
[insertedOrUpdated unionSet: [[notification userInfo] objectForKey: NSInsertedObjectsKey]];
[insertedOrUpdated unionSet: [[notification userInfo] objectForKey: NSUpdatedObjectsKey]];
for (NSManagedObject* foreignMO in insertedOrUpdated)
{
NSManagedObjectID* foreignOID = foreignMO.objectID;
NSManagedObjectID* localOID = [foreignOIDsToLocalOIDs objectForKey: foreignOID];
localOID = localOID ? localOID : foreignOID;
NSManagedObject* localMO = [self objectWithID: localOID];
// Do the attributes.
[localMO setValuesForKeysWithDictionary: [foreignMO dictionaryWithValuesForKeys: [[foreignMO.entity attributesByName] allKeys]]];
// Do the relationships.
NSDictionary* rByName = foreignMO.entity.relationshipsByName;
for (NSString* key in [rByName allKeys])
{
NSRelationshipDescription* desc = [rByName objectForKey: key];
if (!desc.isToMany)
{
NSManagedObject* relatedForeignMO = [foreignMO valueForKey: key];
NSManagedObjectID* relatedForeignOID = relatedForeignMO.objectID;
NSManagedObjectID* relatedLocalOID = [foreignOIDsToLocalOIDs objectForKey: relatedForeignOID];
relatedLocalOID = relatedLocalOID ? relatedLocalOID : relatedForeignOID;
NSManagedObject* localRelatedMO = [self objectWithID: relatedLocalOID];
[localMO setValue: localRelatedMO forKey: key];
}
else
{
id collection = [foreignMO valueForKey: key];
id newCollection = [NSMutableSet set];
if ([collection isKindOfClass: [NSOrderedSet class]])
{
newCollection = [NSOrderedSet orderedSet];
}
for (NSManagedObject* relatedForeignMO in collection)
{
NSManagedObjectID* relatedForeignOID = relatedForeignMO.objectID;
NSManagedObjectID* relatedLocalOID = [foreignOIDsToLocalOIDs objectForKey: relatedForeignOID];
relatedLocalOID = relatedLocalOID ? relatedLocalOID : relatedForeignOID;
NSManagedObject* localRelatedMO = [self objectWithID: relatedLocalOID];
[newCollection addObject: localRelatedMO];
}
[localMO setValue: newCollection forKey: key];
}
}
}
// And delete any objects which pre-existed in my context.
for (NSManagedObject* foreignMO in [[notification userInfo] objectForKey: NSDeletedObjectsKey])
{
NSManagedObjectID* foreignOID = foreignMO.objectID;
NSManagedObject* localMO = [self existingObjectWithID: foreignOID error: NULL];
if (localMO)
{
[self deleteObject: localMO];
}
}
[sourceContext unlock];
}
@end
关于core-data - 立即启用文件NSManagedObjectContext的保存?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9104802/
我尝试为我的 UITableViewController 使用 NSFetchedResultsController(FRC),因为我喜欢 FRC 附带的功能,而不是尝试手动管理我的 UITableV
我的应用程序中目前有 2 个上下文。我的应用程序使用多个选项卡,因此一个选项卡可能正在显示数据,而另一个选项卡可能处于数据输入模式。 我使用一个主要上下文来读取大部分数据以进行显示。当我插入数据时,我
我在 appDelegate 中创建了一个主要的 NSManagedObjectContext。 现在,我正在使用另一个 NSManagedObjectContext 来编辑/添加新对象,而不会影响主
假设我们有一个应用程序需要显示地点列表并且在 3 个线程上运行: 主线程 主线程后台同步(同步有服务器的地方) 地理编码线程(对位置进行地理编码背景) 在所有 3 个线程中,我都有专用的 NSMana
所以我需要使用多线程来使用 cora-data 托管对象上下文。我发现下面的实现几乎是我所需要的。我有 2 个问题: 在哪里释放我分配的 MOC?我不想遍历应用程序中的每个函数线程并调用函数来释放 M
我正在开发一个使用 coredata 的 ipad 应用程序。它下载网络数据库上的信息,并将其记录在 coredata 中。该应用程序基于分割 View 。我的问题是在后台下载和记录数据。 这是我的做
有没有办法在主线程之外的后台保存我的 NSManagedObjectContext ?保存会减慢应用程序的执行速度,因为它通常需要大约 2 秒。 最佳答案 是的,有。 Apple recommends
在 iPhone 的核心数据中,我在尝试将数据保存到 NSManagedObjectContext 时遇到了各种错误。 我相信我的问题都与我使用在多个线程中使用的 NSManagedObjectCon
我有两个实体,每个实体都显示在自己的 UITableView 部分上。 我启用了编辑功能,允许用户通过向右滑动来删除行。这对于第一个实体来说效果很好,但是当我尝试删除第二个实体中的对象时,我收到此错误
我有以下设置: NSManagedObjectContext *parent = [[NSManagedObjectContext alloc]
在调试时,如果我有一个 NSManagedObjectContext,有没有办法查看里面有什么对象。 基本上我有一个保存错误,因为保存了一个不符合 NSCoding 的 CGColor。但我不知道这个
我正在尝试制作一个用户可以编辑 managedObject 属性的应用程序在 View 中,然后选择 Done保留更改,或 Cancel撤消更改。 为了实现此行为,我计划执行以下操作 - 当 View
最近我遇到了一个无法解释的问题。所以我有一个 DocumentContext 类,其中包含一个 Document (NSManagedObject)。这些类的代码是 extension Documen
我想在插入新数据之前清除 CoreData 实体中的所有数据。到目前为止,我尝试过删除所有内容,并且不允许我插入更新的数据,即使对这两个操作使用相同的上下文也是如此。有更好的方法吗?这是我一直在实现的
在我的应用中,我将 Core Data 堆栈从 AppDelegate 移至其自己的名为 LiftEventDataManager 的类中。在一些类中,我获得了对该堆栈的引用,当然,每次我执行操作时都
我正在使用以下代码创建一个对象: +(Checkin *) newCheckinWithId:(NSString*) checkinID forVenueId:(NSString *)venueId
我是 Core Data 的新手。 我的数据模型有一个 User 实体,它有一个属性 fullName,它是从服务器获取的。我使用 NSFetchedResultsController 在 Table
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingSt
我使用 Core Data - 我已经注册并正在监听 NSManagedObjectContextDidSaveNotification:我有一个数据集合(来自 JSON),我想要保存它,在保存所有对
我的核心数据堆栈像往常一样在 AppDelegate 中设置。我是一个优秀的 OO 公民,并认识到可以通过 [[UIApplication sharedApplication] delegate] m
我是一名优秀的程序员,十分优秀!