One of the most frustrating issues that I’ve encountered recently had to do with the relationship between NSManagedObjects and their corresponding NSManagedObjectContext. If you’re unfamiliar with Core Data, an NSManagedObjectContext is a sort of ‘environment’ in which NSManagedObjects can be inserted, fetched, and manipulated. Managed object contexts and the objects that exist within them are closely related, and it doesn’t make much sense to use a managed object without a corresponding context. While it is possible, the object will be a ‘fault’ (in Core Data terms) – that is to say, it will still be a legitimate NSManagedObject with a valid objectID, but all of its attributes will be nil.

I had written some code that would fetch a specific managed object in a temporary context on a background thread, perform some calculations while still in the background, and finally pass the result of these calculations back to the main thread. It seemed like it was working as expected, but then I started seeing some odd behavior when I built and ran on my phone. This wasn’t entirely out of the ordinary, the iOS simulator is much more powerful than my iPhone 4, but generally the only perceptible differences are in performance.

The code was something like this:

- (void)fetchObjectInBackgroundWithCompletion:(void(^)(NSManagedObject *obj))completion{

    dispatch_async(dispatch_get_global_queue(0, 0), ^{

    	// fetch object in temporary background context
        NSManagedObject *obj = [CoreDataUtility fetchObjectFromManagedObjectContext:[self temporaryContext]];

        // do some work with obj...

        // pass the object id to the main thread, and re-fault the object
        NSManagedObjectID *objID = [obj objectID];
        dispatch_async(dispatch_get_main_queue(), ^{
            NSManagedObjectContext *e = [self.managedObjectContext objectWithID:objID];
            completion(e);
        });
    });
}

I ended up seeing issues when manipulating the managed object that was fetched in the temporary background context – for some reason, the object would be a fault, and all its attributes were nil. To confuse matters, the issue was only manifest when I ran with a release configuration on an actual device, if I ran in a debug configuration, or in the simulator, everything seemed to work correctly.

After some debugging, I determined that the issue was with the lifetime of my temporary background context. It turned out that enabling the aggressive compiler optimizations when building an app for release was causing my background context to be deallocated before I could complete the background work with the actual managed objects. Without their parent context, the managed objects were simply faults, and all their attributes were nil, thus producing the confusing results. By simply capturing the temporary context in a local variable until all work with the managed objects was complete, the problem was eliminated.

In retrospect, this was obviously an error on my part – I should have ensured that the lifetime of the context was at least as long as that of the managed objects that had been fetched from it. I did learn that compiler optimization can sometimes have unexpected side effects, so if there are discrepancies between debug and release builds, that’s a good place to start investigating.