objective-c – 将NSArray转换为NSSet,自定义类实例传输不一致

陷入一个有趣的小问题.我正在编写一个方法来过滤数组到唯一对象:
- (NSArray*)distinctObjectsByAddress {
    NSSet* uniqueSet = [NSSet setWithArray:self];
    NSArray* retArray = [uniqueSet allObjects];

    return retArray;
}

并写了一个单元测试来检查:

- (void)testDistinctObjectsByAddress5 {
    Person* adam1 = [[Person alloc] initWithFirstName:@"adam" lastName:@"adam" andParent:nil];
    Person* adam2 = [[Person alloc] initWithFirstName:@"adam" lastName:@"adam" andParent:nil];

    testPersonArray = [NSArray arrayWithObjects:adam1,adam2,nil];

    NSArray* checkArray = [testPersonArray distinctObjectsByAddress];

    STAssertEquals([checkArray count],[testPersonArray count],@"Array %@ counts should match %@ %@",checkArray,adam1,adam2);
}

很简单.有趣的是,测试通过的时间大约有80-90%,并且每次都会失败,因为distinctObjectsByAddress方法只返回一个对象.我已经能够将它跟踪到[NSSet setWithArray:self]调用,但我也能够验证两个人对象是两个不同的对象(至少他们有不同的地址).我假设setWithArray:只是做一个基本的地址比较,但我不明白为什么它有时产生两个像它应该的对象,有时只产生一个.

我刚试过的一件事就是更改adam2,以便名字和名字与adam1不完全相同.这似乎解决错误.当对象在逻辑上相同时,这是否指向某种编译器优化?

解决方法

I’m assuming that setWithArray is just doing a basic address compare

那是不对的. NSSet对添加到它的对象使用-isEqual:和-hash方法.这取决于如何在Person或其超类中实现它们.

如果[person1 isEqual:person2],那么您希望该集合包含一个对象.如果不是,则该集应包含两个对象.

我的猜测是Person在其-isEqual:和-hash方法中没有遵循the rules.最有可能的是,这两个对象是相同的,但它们的哈希值并不像它们应该的那样相等. (除了你幸运的10-20%的时间.)

Does this point to some sort of compiler optimization when the objects are logically the same?

不,没有编译器优化可以将两个对象合并为一个.

相关文章

/** C+⬑ * 默认成员函数 原来C++类中,有6个默认成员函数: 构造函数 析构函数 拷贝...
#pragma once // 1. 设计一个不能被拷贝的类/* 解析:拷贝只会放生在两个场景中:拷贝构造函数以及赋值运...
C类型转换 C语言:显式和隐式类型转换 隐式类型转化:编译器在编译阶段自动进行,能转就转,不能转就编译...
//异常的概念/*抛出异常后必须要捕获,否则终止程序(到最外层后会交给main管理,main的行为就是终止) try...
#pragma once /*Smart pointer 智能指针;灵巧指针 智能指针三大件//1.RAII//2.像指针一样使用//3.拷贝问...
目录<future>future模板类成员函数:promise类promise的使用例程:packaged_task模板类例程...