Подтвердить что ты не робот

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 framework):

- (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 не удаляет первый указатель? Это явно a nil перед вызовом compact.

4b9b3361

Ответ 1

Я видел такое же поведение. Вот отчет об ошибке открытого радара: http://www.openradar.me/15396578

Ответ 2

Причина этого в том, что -compact сначала проверяет, установлен ли внутренний флаг 'needsCompaction'. Если это не так, он просто поручается рано. Единственный раз, когда установлен флаг, является то, что указатель nil вставляется непосредственно в массив через открытый интерфейс. Он не устанавливается, если объект с низкой ссылочной информацией освобождается (и указатель равен nil'd) после того, как указатель был вставлен в массив.

Одна работа для этого поведения заключается в том, чтобы целенаправленно добавлять указатель на массив до вызова -compact. Не идеально, но он будет работать.

[pa addPointer:nil]; // forces the pointer array to do compaction next time
[pa compact];