У меня есть проект, использующий StoryBoards и UISearchDisplayController
, используемый в контексте UINavigationController
, который появляется в корневом контроллере представления. Когда я помещаю новый контроллер представления в стек и вызываю симулированное предупреждение о памяти (или фактически получаю предупреждение о нехватке памяти). Предыдущий контроллер представления успешно выгружает свое представление. Однако, когда я извлекаю второй контроллер представления из стека, я получаю EXC_BAD_ACCESS
. Я включил NSZombies и обнаружил это:
[UISearchDisplayController сохранить]: сообщение отправлено освобожденному экземпляру 0xb13aa30
Я не (по крайней мере, в моем коде) отправляю это сообщение в UISearchDisplayController
. Программно говоря, я ничего с ним не делаю. Точки останова показывают, что я даже не попал в viewDidLoad
первого контроллера представления.
Однако кое-что любопытное: ради смеха и хихиканья я решил полностью retain
SDC в моем viewDidLoad
, просто чтобы посмотреть, что произойдет, и не произойдет ли сбой. Однако мой экземпляр UISearchDisplayController
- это nil
.
Я сделал обратную трассировку и получил этот вывод:
#0 0x01e30e1e in ___forwarding___ ()
#1 0x01e30ce2 in __forwarding_prep_0___ ()
#2 0x01dd1490 in CFRetain ()
#3 0x01eb69c0 in +[__NSArrayI __new::] ()
#4 0x01e0a00a in -[__NSPlaceholderArray initWithObjects:count:] ()
#5 0x01e34f52 in +[NSArray arrayWithObjects:count:] ()
#6 0x01e5e084 in -[NSDictionary allValues] ()
#7 0x01035272 in -[UINib instantiateWithOwner:options:] ()
#8 0x00edce2c in -[UIViewController _loadViewFromNibNamed:bundle:] ()
#9 0x00edd3a9 in -[UIViewController loadView] ()
#10 0x00edd5cb in -[UIViewController view] ()
#11 0x00edd941 in -[UIViewController contentScrollView] ()
#12 0x00eef47d in -[UINavigationController _computeAndApplyScrollContentInsetDeltaForViewController:] ()
#13 0x00eef66f in -[UINavigationController _layoutViewController:] ()
#14 0x00eef93b in -[UINavigationController _startTransition:fromViewController:toViewController:] ()
#15 0x00ef03df in -[UINavigationController _startDeferredTransitionIfNeeded] ()
#16 0x00ef16cb in _popViewControllerNormal ()
#17 0x00ef196c in -[UINavigationController _popViewControllerWithTransition:allowPoppingLast:] ()
#18 0x0b446e82 in -[UINavigationControllerAccessibility(SafeCategory) _popViewControllerWithTransition:allowPoppingLast:] ()
#19 0x00ef0b10 in -[UINavigationController popViewControllerAnimated:] ()
#20 0x00ef297d in -[UINavigationController navigationBar:shouldPopItem:] ()
#21 0x00e7dabe in -[UINavigationBar _popNavigationItemWithTransition:] ()
#22 0x00e7da49 in -[UINavigationBar popNavigationItemAnimated:] ()
#23 0x0b42208c in -[UINavigationBarAccessibility(SafeCategory) popNavigationItemAnimated:] ()
#24 0x00e80507 in -[UINavigationBar _handleMouseUpAtPoint:] ()
#25 0x00e8074c in -[UINavigationBar touchesEnded:withEvent:] ()
#26 0x00e3fa30 in -[UIWindow _sendTouchesForEvent:] ()
#27 0x00e3fc56 in -[UIWindow sendEvent:] ()
#28 0x00e26384 in -[UIApplication sendEvent:] ()
#29 0x00e19aa9 in _UIApplicationHandleEvent ()
#30 0x02d37fa9 in PurpleEventCallback ()
#31 0x01e9e1c5 in __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__ ()
#32 0x01e03022 in __CFRunLoopDoSource1 ()
#33 0x01e0190a in __CFRunLoopRun ()
#34 0x01e00db4 in CFRunLoopRunSpecific ()
#35 0x01e00ccb in CFRunLoopRunInMode ()
#36 0x02d36879 in GSEventRunModal ()
#37 0x02d3693e in GSEventRun ()
#38 0x00e17a9b in UIApplicationMain ()
#39 0x00002b72 in main (argc=1, argv=0xbffff620)
Кажется, там нет ничего действительно интересного (есть ли когда-нибудь? :P), и, похоже, это все внутренности Apple. Любые идеи о том, как заставить эту проблему уйти?
ОБНОВЛЕНИЕ: даже когда я удаляю связь между моим контроллером представления и свойством контроллера отображения поиска, но создаю для него собственный IBOutlet
, он все равно дает сбой. Может плохой баг?
ОБНОВЛЕНИЕ 2: когда я программно создаю собственный экземпляр UISearchDisplayController
(не через раскадровку) и создаю его в viewDidLoad
, все работает так, как должно.
ОБНОВЛЕНИЕ 3: я могу постоянно воспроизводить эту проблему в новом проекте с раскадровкой. Я сделал то же самое с ванильным пером, и все заработало так, как предполагалось. Однако, если я настрою то же самое, используя раскадровку и переход, он взорвется, как в моем реальном проекте. :(
RECAP. Вот шаги по воссозданию этой проблемы:
- Создайте контроллер представления в раскадровке с помощью
UISearchDisplayController
- Вставьте новый контроллер представления в стек навигации.
- Причина предупреждения о нехватке памяти
- Вытолкнуть контроллер из стека
- КАБУМ!
В этот момент viewDidLoad
даже не вызывается на первом контроллере представления, код Apple взрывается до этого.
IBOutlets
и вызываю реализацию моего супераviewDidUnload
. - person Wayne Hartman   schedule 20.12.2011