NSPointerArray странное уплотнение
У меня слабый NSPointerArray
с некоторыми NSObject
это было выпущено. Перед звонком compact
что я вижу это:
(lldb) po [currentArray count]
1
(lldb) po [currentArray pointerAtIndex:0]
<nil>
(lldb) po [currentArray allObjects]
<__NSArrayM 0x16f04f00>(
)
Это имеет смысл, но что действительно странно, так это то, что когда я звоню compact
в этом массиве я вижу те же значения! Граф все еще возвращает 1 и pointerAtIndex:0
является nil
,
Почему ноль не был удален?
РЕДАКТИРОВАТЬ
Вот полный код (да, это фреймворк XCTesting):
- (void)testCompaction {
__weak id testingPointer = nil;
NSPointerArray *weakArray = [NSPointerArray weakObjectsPointerArray];
@autoreleasepool {
NSObject *someObj = [[NSObject alloc] init];
testingPointer = someObj;
[weakArray addPointer:(__bridge void*)testingPointer];
NSLog(@"before compaction inside autorelease: testingPointer = %@ count = %d, allObjects = %@, pointerAtIndex:0 = %@, pointerAtIndex:0 class = %@", testingPointer, [weakArray count], [weakArray allObjects], [weakArray pointerAtIndex:0], [(id)[weakArray pointerAtIndex:0] class]);
someObj = nil;
}
NSLog(@"before compaction outside autorelease: testingPointer = %@ count = %d, allObjects = %@, pointerAtIndex:0 = %@, pointerAtIndex:0 class = %@", testingPointer, [weakArray count], [weakArray allObjects], [weakArray pointerAtIndex:0], [(id)[weakArray pointerAtIndex:0] class]);
[weakArray compact];
NSLog(@"after compaction outside autorelease: testingPointer = %@ count = %d, allObjects = %@, pointerAtIndex:0 = %@, pointerAtIndex:0 class = %@", testingPointer, [weakArray count], [weakArray allObjects], [weakArray pointerAtIndex:0], [(id)[weakArray pointerAtIndex:0] class]);
}
и журналы:
before compaction inside autorelease: testingPointer = <NSObject: 0x7de7ff80> count = 1, allObjects = (
"<NSObject: 0x7de7ff80>"
), pointerAtIndex:0 = <NSObject: 0x7de7ff80>, pointerAtIndex:0 class = NSObject
2015-07-20 14:27:14.062 AppetizeSuite copy[54144:9019054] before compaction outside autorelease: testingPointer = (null) count = 1, allObjects = (
), pointerAtIndex:0 = (null), pointerAtIndex:0 class = (null)
2015-07-20 14:27:22.615 AppetizeSuite copy[54144:9019054] after compaction outside autorelease: testingPointer = (null) count = 1, allObjects = (
), pointerAtIndex:0 = (null), pointerAtIndex:0 class = (null)
Почему compact
метод не удаляет первый указатель? Это явно nil
перед звонком compact
,
2 ответа
Я видел такое же поведение. Вот отчет об ошибке открытого радара: http://www.openradar.me/15396578
Это происходит потому, что -compact
Сначала проверяется, установлен ли внутренний флаг needsCompaction. Если это не так, он просто освобождается под залог рано. Единственный раз, когда устанавливается флаг, это если нулевой указатель вставляется непосредственно в массив через открытый интерфейс. Он не устанавливается, если объект со слабой ссылкой освобожден (и указатель равен nil'd) после того, как указатель был вставлен в массив.
Одним из способов обойти это поведение является целенаправленное добавление нулевого указателя в массив перед вызовом. -compact
, Не идеально, но это сработает.
[pa addPointer:nil]; // forces the pointer array to do compaction next time
[pa compact];
Прошло уже несколько лет, и использование примера кода не показывает изменений.
Однако -compact отлично работает, когда обновление указателя выполняется с помощью метода -replacePointerAtIndex:withPointer:
В приведенном выше коде замена
someObj = nil;
по
[weakArray replacePointerAtIndex:0 withPointer:nil];
обеспечивает ожидаемый результат после [weakArray compact];
и код выйдет из строя при последнем NSLog