OC中自定义类的NSCopying实现的注意事项(isEqual & hash实现)

在OC中,如果自定义类,则要考虑赋值、持久化保存、保存到其它容器中等各种情况的对象复制和比较,下面是一个比较全面的自定义例子,在此仅作记录:

自定义类:

KeyValuePairs.h:

#import <Foundation/Foundation.h>

@interface KeyValuePairs: NSObject <NSCopying>
@property (nonatomic,strong)NSString *identifier;
@property (nonatomic,strong)NSString *name;

@end

KeyValuePairs.m:

#import "KeyValuePairs.h"

@implementation KeyValuePairs

- (id)copyWithZone:(NSZone *)zone
{
    KeyValuePairs *kvp = [[[self class] allocWithZone:zone] init];
    kvp.identifier = self.identifier;
    kvp.name = self.name;
    return kvp;
}

- (BOOL)isEqualToKeyValuePairs:(KeyValuePairs *)kvp{
    if (!kvp) {
        return NO;
    }
    BOOL haveEqualName = (!self.name && !kvp.name) || [self.name isEqualToString:kvp.name];
    BOOL haveEqualIdentifier = (!self.identifier && !kvp.identifier) || [self.identifier isEqualToString:kvp.identifier];
    
    return haveEqualName && haveEqualIdentifier;
}

#pragma mark -NSObject
-(BOOL)isEqual:(id)object{
    if (self == object) {
        return YES;
    }
    if (![object isKindOfClass:[KeyValuePairs class]]) {
        return NO;
    }
    return [self isEqualToKeyValuePairs:(KeyValuePairs *)object];
}

- (NSUInteger)hash {
    return [self.name hash] ^ [self.identifier hash];
}

@end


测试:

    NSMutableDictionary *namesWillUpdateDic = [[NSMutableDictionary alloc] init];
    NSMutableArray *names = [[NSMutableArray alloc] init];
    for (int i = 0; i<1000; i++) {
        NSString *name = [NSString stringWithFormat:@"%d_zhangsan",i];
        NSString *identifier = [NSString stringWithFormat:@"%d_identifier",i];
        NSString *strObj = [NSString stringWithFormat:@"%d_strObj",i];
        KeyValuePairs *kvp = [[KeyValuePairs alloc] init];
        kvp.identifier = identifier;
        kvp.name = name;
        [namesWillUpdateDic setObject:strObj forKey:kvp];
        [names addObject:kvp];
    }
    
    for (int j = 0; j<1000; j++) {
        int index = arc4random()%1000;
        KeyValuePairs *kvp = [names objectAtIndex:index];
        NSString *strObj = [namesWillUpdateDic objectForKey:kvp];
        NSString *msg = [NSString stringWithFormat:@"index:%d,identifier:%@,email:%@,strObj:%@",index,kvp.identifier,kvp.name,strObj];
        NSLog(@"%@",msg);
    }


注意:

1、如果自定义类不实现NSCopying则无法进行持久化保存(到文件)、或错误赋值

2、如果自定义类不重写isEqual则默认使用内存地址比较两个对象,可能会出现意想不到的结果

3、isEqual和hash方法要同时重写,否则isEqual方法判断将不正确

在 Objective-C ,深拷贝(Deep Copy)和浅拷贝(Shallow Copy)是用于复制对象的两种不同方式。 浅拷贝是指创建一个新的对象,该对象与原始对象共享数据的内存地址。换句话说,新对象只是原始对象的一个引用,对新对象的修改也会影响原始对象。在 Objective-C ,可以使用 `copy` 方法来执行浅拷贝。 深拷贝是指创建一个新的对象,并且复制原始对象的所有数据。这意味着新对象有自己的内存地址,对新对象的修改不会影响原始对象。在 Objective-C ,可以使用 `mutableCopy` 方法来执行深拷贝。 需要注意的是,深拷贝只会复制对象本身,而不会递归地复制对象所包含的其他对象。如果需要对对象的所有数据进行递归复制,可以通过实现 NSCopying 协议来自定义深拷贝操作。 下面是一个示例代码,演示了如何执行浅拷贝和深拷贝: ```objective-c #import <Foundation/Foundation.h> @interface Person : NSObject <NSCopying> @property (nonatomic, copy) NSString *name; @end @implementation Person - (instancetype)copyWithZone:(NSZone *)zone { Person *copy = [[Person allocWithZone:zone] init]; copy.name = self.name; return copy; } @end int main(int argc, const char * argv[]) { @autoreleasepool { Person *person1 = [[Person alloc] init]; person1.name = @"John"; // 浅拷贝 Person *person2 = [person1 copy]; NSLog(@"person1: %@, person2: %@", person1.name, person2.name); // 输出:person1: John, person2: John // 深拷贝 Person *person3 = [person1 mutableCopy]; person3.name = @"Emily"; NSLog(@"person1: %@, person3: %@", person1.name, person3.name); // 输出:person1: John, person3: Emily } return 0; } ``` 在上面的示例,使用 `copy` 方法执行了浅拷贝操作,`mutableCopy` 方法执行了深拷贝操作。注意,为了实现深拷贝,我们在 `Person` 遵循了 NSCopying 协议,并重写了 `copyWithZone:` 方法自定义深拷贝操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值