I am using Core Data to model an entity which has both attributes and relationships. I would like to make one of the attributes dependent on two other relationships.
The Core Data FAQ and several other examples use +(NSSet *)keyPathsForValuesAffectingValueForKey:(NSString *)key
for this purpose. However, this only seems to be called for all of my relationship properties, but not for any of the attributes. In addition keyPathsForValuesAffecting<key>
never gets called for any of my attributes.
Is this the normal behaviour or am I missing something? How can I calculate one attribute based on other attributes or properties, eg. a distance, by setting a startLocation and a stopLocation?
It looks like a found the solution to my own problem after a couple of days and it turns out I was missing something.
After rereading the discussion of the method
keysPathsForValuesAffectingValueForKey:
in the NSKeyValueObserving Protocol Reference, I realized the meaning of the following sentence:In short, your instance should have an observer observing changes to your attribute
<key>
:As soon as you have an observer registered, the protocol will call
keysPathsForValuesAffectingValueForKey
for your specific attribute key. If this method returns a non-empty set of key paths, KVO will emit a change notification for your attribute, if a change is made to any of these key paths, in addition to notifying you of any direct change to your attribute.Relationship keys do get called automatically, because Core Data already uses observers to keep inverse relationships up to date.
In the particular case where you want to have an attribute depend on another attribute or relationship within the same entity, you will have to:
addObserver:forKeyPath:options:context:
keyPathsForValuesAffectingValueForKey:
orkeyPathsForValuesAffecting<key>
observeValueForKeyPath:ofObject:change:context
for your attribute key path to act on the relevant change notifications, i.e. updating your attribute value.