ios - 有没有办法以通用方式替换关系的添加和删除方法的核心数据实现?

标签 ios core-data nsmanagedobject

请注意:这与 this question 不是同一个问题,因为我已经知道这源于 Apple 错误。请注意,覆盖特定方法(例如,-addFriendsObject:,如果我有 friend 关系)不是一种选择,因为我需要在类别中执行此操作,以便它适用于任何托管对象,而不管修改模型和重建从中自动生成类。

对此的需求源于这样一个事实,即显然在使关系有序化的那一刻,对多(NSMutableOrderedSets)核心数据动态方法就会陷入困境:

  1. 方法 -add<Relationship>Object , -add<Relationship> , -remove<Relationship>Object-remove<Relationship>都会崩溃,异常相似 'NSInvalidArgumentException', reason: '*** -[NSSet intersectsSet:]: set argument is not an NSSet'
  2. 按照 insertObject:in<Relationship>AtIndex: 的方法反而会崩溃,因为 CoreData 实际上没有为它们提供任何实现。

至于2,我写了一个覆盖-methodSignatureForSelector:的类别和 -forwardInvocation而是做类似 mutableOrderedSetValueForKey 的事情对于关系名称,后跟实际的添加或删除。

现在对于 1,问题是 CoreData 实际上为这些方法提供了实现(尽管它们不是有序集的正确实现)。所以我也需要一种拦截这些选择器的方法,这样我就可以通过 mutableOrderedSetValueForKey 实现行为。

有什么想法可以实现吗?

最佳答案

我不知道这是最好的方法(它肯定不是有史以来最漂亮的方法),但它完成了工作:

#import "NSManagedObject+OrderedSets.h"
#import <objc/runtime.h>

@implementation NSManagedObject (OrderedSets)

+ (void)swizzleMethod:(SEL)originalSelector with:(SEL)replacementSelector
{
    const char *methodTypeEncoding = method_getTypeEncoding(class_getInstanceMethod([self class], originalSelector));
    class_replaceMethod(self,
                        originalSelector,
                        class_getMethodImplementation(self, replacementSelector),
                        methodTypeEncoding);
}

+ (void)initialize
{
    NSEntityDescription *selfEntity = // get a hold of your NSManagedObjectContext and from its entities grab model.entitiesByName[NSStringFromClass(self)] ... 
    for (NSString *rKey in [selfEntity relationshipsByName]) {
        NSRelationshipDescription *r = selfEntity.relationshipsByName[rKey];
        if (r.isOrdered) {
            NSString *rKeyFirstCaps = [[rKey substringToIndex:1] capitalizedString];
            NSString *capitalizedKey = [rKey stringByReplacingCharactersInRange:NSMakeRange(0,1) withString:rKeyFirstCaps];
            NSString *addSelectorString = [NSString stringWithFormat:@"add%@Object:", capitalizedKey];
            NSString *removeSelectorString = [NSString stringWithFormat:@"remove%@Object:", capitalizedKey];
            [self swizzleMethod:NSSelectorFromString(addSelectorString) with:@selector(addOrRemoveObjectInOrderedSet:)];
            [self swizzleMethod:NSSelectorFromString(removeSelectorString) with:@selector(addOrRemoveObjectInOrderedSet:)];
        }
    }
}

- (void)addOrRemoveObjectInOrderedSet:(NSManagedObject*)object
{
    NSString *selectorString = NSStringFromSelector(_cmd);
    NSString *prefix = [selectorString hasPrefix:@"add"] ? @"add" : @"remove";
    selectorString = [selectorString stringByReplacingCharactersInRange:[selectorString rangeOfString:prefix] withString:@""];
    selectorString = [selectorString stringByReplacingCharactersInRange:[selectorString rangeOfString:@"Object" options:NSBackwardsSearch] withString:@""];
    selectorString = [selectorString substringToIndex:selectorString.length - 1];
    NSString *selectorFirstLowerCase = [[selectorString substringToIndex:1] lowercaseString];
    NSString *camelCasedKey = [selectorString stringByReplacingCharactersInRange:NSMakeRange(0,1) withString:selectorFirstLowerCase];

    if ([prefix isEqualToString:@"add"]) {
        [[self mutableOrderedSetValueForKey:camelCasedKey] addObject:object];
    }
    else {
        [[self mutableOrderedSetValueForKey:camelCasedKey] removeObject:object];
    }
}

@end

关于ios - 有没有办法以通用方式替换关系的添加和删除方法的核心数据实现?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16805624/

相关文章:

iphone - iTunes 无法从同步服务加载数据类信息

ios - 如何将核心数据添加到现有的 Xcode 9 Swift 4 iOS 11 项目中?

objective-c - 将 NSManagedObjectModel 中的关系添加到以编程方式创建的 NSEntityDescription

ios - 核心数据|与 2 个逆元的关系

objective-c - 核心数据持久存储中的物理位置(目录)( Objective-C )

ios - 如何列出 NSManagedObject 的变量

objective-c - 从位于自定义 UITableViewCell 中的 UITextView 获取文本

ios - Realm:在编写新对象时,如何避免为关系创建对象?

ios - 如何过渡到带有导航栏的 View Controller ?

ios - Core Data 创建新对象 Swift