我知道在Objective-C中正常的NSArray串联是如何工作的。这不是那个问题。

我有正在从Web服务增量更新的数据。我的对象具有以下类定义(已删除很多内容):

//  NoteTemplate.h
#import <UIKit/UIKit.h>
@interface NoteTemplate

@property (copy, nonatomic) NSString *objectId;


我正在缓存这些设备上的列表,并在启动时检查以查看数据库中是否有任何新的或更新的NoteTemplate对象要加载。因此,我最终得到两个数组:

NSArray <NoteTemplate *> *oldArray
NSArray <NoteTemplate *> *newArray


如果没有更新,那么我要做的就是简单地将两个数组连接在一起,就是这样。

但是,如果有更新,我想合并两个数组,但是只要有共同的objectId,则newArray中的项目应优先于oldArray中的项目。

到目前为止,我是这样粗暴执行的:

- (void)updateNoteTemplatesWithArray:(NSArray *)newTemplates {
    NSArray *oldTemplates = [self getNoteTemplates];
    NSMutableArray *combined = [NSMutableArray arrayWithArray:newTemplates];
    for (NoteTemplate *noteTemplate in oldTemplates) {
        NSArray *matches = [combined filteredArrayUsingPredicate:[NSPredicate predicateWithBlock:^BOOL(id blockTemplate, NSDictionary<NSString *,id> *bindings) {
            return [((NoteTemplate *)blockTemplate).objectId isEqualToString:noteTemplate.objectId];
        }]];
        if (matches.count == 0) {
            [combined addObject:noteTemplate];
        }
    }
    [self setNoteTemplates:[combined copy]];
}


有没有更优化的方法来做到这一点?我完全看不到这会影响性能,因此也许没有必要进行优化。尽管如此,这种方法还是很笨拙,而且设计过度。

最佳答案

要使用Set用法扩展@Larme的建议,您可以尝试以下方法:

@interface NoteTemplate: NSObject

@property (copy, nonatomic) NSString *objectId;
@property (copy, nonatomic) NSString *text;

- (instancetype)initWithObjectId:(NSString *)objectId text:(NSString *)text;

@end

@implementation NoteTemplate
- (instancetype)initWithObjectId:(NSString *)objectId text:(NSString *)text {
    self = [super init];
    if (self != nil) {
        _objectId = objectId;
        _text = text;
    }
    return self;
}

- (BOOL)isEqual:(id)object {
    return [self.objectId isEqualToString:[object objectId]];
}

@end


以及使用代码:

NoteTemplate *nt1 = [[NoteTemplate alloc] initWithObjectId:@"1" text:@"old set"];
NoteTemplate *nt2 = [[NoteTemplate alloc] initWithObjectId:@"2" text:@"old set"];
NoteTemplate *nt3 = [[NoteTemplate alloc] initWithObjectId:@"1" text:@"new set"];
NoteTemplate *nt4 = [[NoteTemplate alloc] initWithObjectId:@"3" text:@"new set"];

NSSet <NoteTemplate *> *oldSet = [NSSet setWithObjects:nt1, nt2, nil];
NSSet <NoteTemplate *> *newSet = [NSSet setWithObjects:nt3, nt4, nil];

NSMutableSet <NoteTemplate *> *mergedSet = [newSet mutableCopy];
[mergedSet unionSet:oldSet];

for (NoteTemplate *note in mergedSet) {
    NSLog(@"Set item %@ %@", note.objectId, note.text);
}


执行此代码后,您将在日志中看到:

Set item 3 new set
Set item 1 new set
Set item 2 old set


我认为这就是您想要的。

关于ios - Objective-C:有选择地组合两个数组,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/48771147/

10-10 21:05