Как определить, какой NSNotification падает из-за освобождения наблюдателя - PullRequest
10 голосов
/ 19 февраля 2012

Золотое правило использования NSNotification похоже на

"вызов removeObserver до освобождения observer (или object)".

Я имею дело с кодовой базой, где это правило не соблюдалось, но я не могу определить местонахождение нарушения. Я искал код и убедился, что у каждого addObserver есть соответствующий removeObserver, но я все еще вижу сообщения о сбоях следующего вида:

OS Version:      iPhone OS 5.0.1 (9A405)
Report Version:  104

Exception Type:  SIGSEGV
Exception Codes: SEGV_ACCERR at 0x8
Crashed Thread:  0

Thread 0 Crashed:
0   libobjc.A.dylib                     0x31516fbc objc_msgSend + 16
1   Foundation                          0x3195b50f __57-[NSNotificationCenter addObserver:selector:name:object:]_block_invoke_0 + 19
2   CoreFoundation                      0x37a02577 ___CFXNotificationPost_block_invoke_0 + 71
3   CoreFoundation                      0x3798e0cf _CFXNotificationPost + 1407
4   Foundation                          0x318cf3fb -[NSNotificationCenter postNotificationName:object:userInfo:] + 67
5   UIKit                               0x34e5ee25 -[UIApplication _handleApplicationSuspend:eventInfo:] + 697
6   UIKit                               0x34deed17 -[UIApplication handleEvent:withNewEvent:] + 2031
7   UIKit                               0x34dee3bf -[UIApplication sendEvent:] + 55
8   UIKit                               0x34dedd2d _UIApplicationHandleEvent + 5809
9   GraphicsServices                    0x3750bdf3 PurpleEventCallback + 883
10  CoreFoundation                      0x37a0a553 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__ + 39
11  CoreFoundation                      0x37a0a4f5 __CFRunLoopDoSource1 + 141
12  CoreFoundation                      0x37a09343 __CFRunLoopRun + 1371
13  CoreFoundation                      0x3798c4dd CFRunLoopRunSpecific + 301
14  CoreFoundation                      0x3798c3a5 CFRunLoopRunInMode + 105
15  GraphicsServices                    0x3750afcd GSEventRunModal + 157
16  UIKit                               0x34e1c743 UIApplicationMain + 1091
17  App                                 0x00002d2f main (main.m:14)

Моя интерпретация этого отчета о сбое заключается в том, что [UIApplication _handleApplicationSuspend:eventInfo:] публикует уведомление, для которого наблюдатель был освобожден перед удалением.

Предполагая, что эти интерпретации верны, как мне определить, какое уведомление публикуется ? И в идеале, что такое освобожденный тип объекта?

Ответы [ 2 ]

6 голосов
/ 19 февраля 2012

Вы можете установить символическую точку останова в -[NSNotificationCenter postNotificationName:object:userInfo:] и распечатать третий переданный ей аргумент (первый - NSNotificationCenter, второй - _cmd), используя команду отладчика po.

0 голосов
/ 26 марта 2013

Другой способ определить это - получить значение параметра name (строку, которую вы передали для публикации уведомления) и убедиться, что любой объект, который наблюдает за этой строкой / именем, настроен на удаление наблюдения когда-нибудь в жизненном цикле объектов или как таковое. освобождает.

...