我在标签栏应用程序中遇到了一个奇怪的问题,看起来像是过度发布。对于这个问题描述的复杂性,我提前道歉。希望一些示例代码会有所帮助。
我将我的应用程序委托(delegate) 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];
}
}
标签栏 View 为 5 个标签中的每一个设置了一个
UINavigationController
。 tab 4 配置的 CrashingViewController
中的根 View Controller (称为 UINavigationController
)源自 UIViewController
并符合 UITableViewDataSource
和 UITableViewDelegate
协议(protocol)以支持 UITableView
subview ,它只是一个 4 行表,其中每个单元格都允许用户导航到另一个 View 。在 -[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
的 View 。然后,用户点击表格中的第二个单元格,将 ASubViewController2
的实例推送到导航堆栈上。到现在为止还挺好。现在,如果用户点击选项卡栏上的另一个选项卡,则会显示另一个 View Controller 的 View 。如果用户然后再次点击第 4 个选项卡,配置为该选项卡的根 View Controller 的 UINavigationController
弹出其所有项目,并应该显示其根 View Controller 。正是在此期间,应用程序崩溃了,因为一路上的 UIKit 框架在 svc2
上的某个地方调用了 [ASubViewController2 RespondsToSelector:] 在它已经被释放之后。我遇到困难的事情是:为什么当用户导航到
ASubViewController2
时会发生这种崩溃,而其他 ASubViewController*
却没有?此外,如果我未能释放 svc2
,则不会发生崩溃 - 所以这绝对是 svc2
某处的双重释放,我只是不知道发生了什么。 svc2
是一个方法局部变量,我在分配它后立即释放它。当 UITableViewController
从导航 Controller 的堆栈中弹出时,UIViewController
的行为是否与 dealloc
有所不同?我还缺少其他东西吗?我确定我只是遗漏了一些有据可查的或其他直接的东西,但我已经研究了足够长的时间以找到一些盲点。
更新 :
我确实启用了 NSZombies。这是控制台中给出的消息:
*** -[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 :
ASubViewController2
的 svc2
方法在用户第二次点击 tab 4 之后,僵尸消息输出之前,在 ojit_code 上调用。我的代码都不在 dealloc 的代码路径中。 最佳答案
我遇到了类似的问题。似乎 View Controller 已经被释放,但它的表 View 仍然调用被释放的 View Controller 的委托(delegate)。只需在 View Controller 的 dealloc 中添加以下或类似的行:
self.tableView.delegate = nil;
self.tableView.dataSource = nil;
它对我有帮助!
关于iphone - 从 UINavigationController 堆栈中弹出时 UITableViewController 是否有特殊行为?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/3740502/