我在标签栏应用程序中有一个奇怪的似乎是一个过度发布的问题。我对此问题描述的复杂性表示歉意。希望一些示例代码可以提供帮助。
我已将我的应用程序委托MyAppDelegate
设置为UITabBarControllerDelegate
:
- (BOOL)tabBarController:(UITabBarController *)tabBarControllerIn shouldSelectViewController:(UIViewController *)viewController {
return YES;
}
- (void)tabBarController:(UITabBarController *)tabBarControllerIn didSelectViewController {
if ([viewController isKindOfClass:[UINavigationController class]]) {
UINavigationController *navController = (UINavigationController *)viewController;
[navController popToRootViewControllerAnimated:NO];
}
}
标签栏视图的设置为UINavigationController
,每个标签都有5个标签。配置为选项卡4的CrashingViewController
中的根视图控制器(称为UINavigationController
)派生自UIViewController
,并符合UITableViewDataSource
和UITableViewDelegate
协议以支持一个UITableView
子视图,它只是一个4行表,每个单元格允许用户导航到另一个视图。在-[UITableViewDelegate tableView:didSelectRowAtIndexPath:]
:
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
ASubViewController1 *svc1;
ASubViewController2 *svc2; // this one inherits from UITableViewController;
ASubViewController3 *svc3;
ASubViewController4 *svc4;
switch (indexPath.row) {
case 1:
svc1 = [[ASubViewController1 alloc] init];
[self.navigationController pushViewController:svc1 animated:YES];
[svc1 release];
break;
case 2:
svc2 = [[ASubViewController2 alloc] init];
[self.navigationController pushViewController:svc2 animated:YES];
[svc2 release]; // if I comment this out, I avoid the problem described
break;
/* case 3 and 4 are exactly like the previous two */
default:
break;
}
}
如果用户选择标签栏的第4个标签,则会显示CrashingViewController
的视图。然后,用户点击其表中的第二个单元格,该表格将ASubViewController2
的实例推送到导航堆栈。到现在为止还挺好。现在,如果用户点击选项卡栏上的另一个选项卡,则会显示另一个视图控制器的视图。如果用户再次点击第4个选项卡,则配置为该选项卡的根视图控制器的UINavigationController
将弹出其所有项目,并且应显示其根视图控制器。在此过程中应用程序崩溃了,因为沿途的UIKit框架在svc2
已经被释放后调用[ASubViewController2 respondsToSelector:]。
我遇到困难的是:为什么当用户导航到ASubViewController2
时会发生此崩溃,但其他ASubViewController*
都没有?此外,如果我无法释放svc2
,则不会发生崩溃 - 因此它绝对是某个地方svc2
的双重释放,我只是不知道发生了什么。 svc2
是一个方法局部变量,我在分配它后立即释放它。从UITableViewController
弹出导航控制器的堆栈时,UIViewController
以某种方式表现不同吗?我错过了别的什么吗?
我确信我只是遗漏了一些记录良好或其他直截了当的东西,但我一直在寻找足够长的时间来找到一些盲点。
更新:
我确实有NSZombiesEnabled。这是控制台中给出的消息:
*** -[ASubViewController2 respondsToSelector:]: message sent to deallocated instance 0xe345cd0
回溯:
#0 0x02a18fa7 in ___forwarding___ ()
#1 0x02a18e72 in __forwarding_prep_0___ ()
#2 0x003e69be in -[UITableView(UITableViewInternal) _spacingForExtraSeparators] ()
#3 0x003ede94 in -[UITableView(_UITableViewPrivate) _adjustExtraSeparators] ()
#4 0x003e6743 in -[UITableView layoutSubviews] ()
#5 0x027b9481 in -[CALayer layoutSublayers] ()
#6 0x027b91b1 in CALayerLayoutIfNeeded ()
#7 0x027b22e0 in CA::Context::commit_transaction ()
#8 0x027b2040 in CA::Transaction::commit ()
#9 0x027e2ebb in CA::Transaction::observer_callback ()
#10 0x02a88f4b in __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__ ()
#11 0x02a1db27 in __CFRunLoopDoObservers ()
#12 0x029e6ce7 in __CFRunLoopRun ()
#13 0x029e6350 in CFRunLoopRunSpecific ()
#14 0x029e6271 in CFRunLoopRunInMode ()
#15 0x0329200c in GSEventRunModal ()
#16 0x032920d1 in GSEventRun ()
#17 0x00380af2 in UIApplicationMain ()
#18 0x0000226a in main (argc=1, argv=0xbfffef64) at ~/main.m:16
更新2 :
在用户第二次点击标签4并输出僵尸消息之前,在dealloc
上调用ASubViewController2
的{{1}}方法。我的所有代码都不在dealloc的代码路径中。
答案 0 :(得分:5)
self.tableView.delegate = nil; self.tableView.dataSource = nil;
对我有帮助!