照著網上的教程,吭哧吭哧寫好了一個tableView,Run起來,界面出現了。還來不及高興一下,一滾動,坑爹啊,直接crash掉了,退出。
打開調試模式,再重現一次,Debugger Console顯示的結果是“Program received signal: "EXC_BAD_ACCESS"”.
這個結果表示代碼中有數組越界或者哪里內存泄漏了。(網上教程不靠譜啊,完全照著教程敲代碼竟然都錯了)
還好總共沒幾個方法,多加幾個斷點調試一下。
問題出來了:
當向下滾動,調用cellForRowAtIndexPath消息時,執行到第25行代碼tableData已經被回收了。
問題分析:
數組初始化放在viewDidLoad消息中,照著教程敲使用
arrayWithObjects來創建。從語法上講,此方法是autorelease的,不需要我們手動去release。
然而問題也出在這,當向下滾動時,不知為何在viewDidLoad初始化的數組都已經被回收了。
修正方案:
把viewDidLoad消息中數組創建方法都改為[
[NSArray alloc] initWithObject: ……];方式創建,再在dealloc中釋放掉就OK了。
推斷總結:
此代碼的界面是用IB拉出來的,對于新人來說我們并不清楚它的view的創建細節。從執行效果來看,它在創建view時嵌套了個NSAutoreleasePool,向下滾動時已經把pool給release掉了,所以出現crash。
得到教訓就是:使用IB創建界面時,那些自定義的對象盡量使用alloc/retain/copy方式創建,自己release。把內存管理寄托在自動釋放池中是不靠譜的,除非這個池是你自己創建并釋放的。
源代碼如下
1
2 - (void)viewDidLoad {
3 [super viewDidLoad];
4 tableData = [NSArray arrayWithObjects:@"香辣醬香骨",@"肉末燒豆腐",@"芙蓉蝦",@"紅燒帶魚",
5 @"粉蒸排骨",@"自來紅月餅",@"蛋黃蓮蓉月餅",@"南瓜奶酪蛋糕",
6 @"南瓜沙拉",@"五香毛豆",@"奶油冰淇淋",@"焦糖南瓜冰淇淋",nil];
7 thumbnails =[NSArray arrayWithObjects:@"a.jpg",@"b.jpg",@"c.jpg",@"d.jpg",@"e.jpg",@"f.jpg",
8 @"g.jpg",@"h.jpg",@"i.jpg",@"j.jpg",@"k.jpg",@"l.jpg",nil];
9 prepTime = [NSArray arrayWithObjects:@"10 min",@"5 min",@"5 min",@"10 min",@"8 min",@"30 min",
10 @"30 min",@"45 min",@"20 min",@"5 min",@"50 min",@"40 min",nil];
11 }
12
13 - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
14 return [tableData count];
15 }
16
17 - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
18 static NSString *simpleTableIdentifier = @"SimpleTableItem";
19 SimpleTableCell *cell = (SimpleTableCell *)[tableView dequeueReusableCellWithIdentifier:simpleTableIdentifier];
20 if(cell == nil){
21 NSArray *nib = [[NSBundle mainBundle] loadNibNamed:@"SimpleTableCell" owner:self options:nil];
22 cell = [nib objectAtIndex:0];
23 }
24 NSUInteger row = [indexPath row];
25 cell.nameLabel.text = [tableData objectAtIndex:row];
26 cell.thumbnailImageView.image = [UIImage imageNamed:[thumbnails objectAtIndex:row]];
27 cell.prepTimeLabel.text = [prepTime objectAtIndex:row];
28 return cell;
29 }
30
31 - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
32 return 78;
33 }
34
35 - (void)dealloc {
36 [super dealloc];
37 }