iOS数据、界面分开设计模式遇到的一个问题

来源:互联网 发布:sql join on 多表 编辑:程序博客网 时间:2024/06/01 10:02
我们习惯在开发中把数据和界面分开实现,这种方式比较好,只需要在数据和界面中同时依赖一个数据结构即可,这种做法对于解藕是一个不错的方式。

但是有一些细节的地方可能会导致我们遇到一些很难查找的bug,比如我们之前遇到的一个问题,现在分享给大家。

先来描述一下问题:我们在UITableView中加入了一个向下拖动刷新数据的控件,控件是EGORefreshTableHeaderView。拖动后,我们就使用ASIHttpRequest刷新数据,但是在拖动幅度大一些时,ASIHttpRequest请求发出后就直接崩溃了,而且看栈也看不出崩在哪。

数据请求代码如下:

- (void)startGetNewsListData{    //newsDataArray是一个成员变量,用于取到数据后用于postNotificationName,给UITableView使用到    if (nil !=newsDataArray &&newsDataArray.count >0)    {        [newsDataArrayremoveAllObjects];    }    NSString *strURL = @"*****";//此处需要填入url的地址字符串    NSURL *url = [NSURLURLWithString:strURL];    ASIHTTPRequest *request = [ASIHTTPRequestrequestWithURL:url];    //设定委托,委托自己实现异步请求方法    [request setDelegate : self ];    // 开始异步请求    [requeststartAsynchronous ];//执行完这句后,就直接崩溃了    //[request release];}- ( void )requestFinished:( ASIHTTPRequest *)request{    NSString *strRequest = [request responseString];    SBJsonParser *parser = [[SBJsonParseralloc]init];      NSDictionary *json = [parser objectWithString:strRequest error:nil];    int nCounts = [[json objectForKey:@"counts"]intValue];    NSArray *activities = [json objectForKey:@"news"];     for (int i =0; i< nCounts; i++)    {        NSDictionary *dictSummary = [activitiesobjectAtIndex:i];        NewsData *data = [[NewsDataalloc]init];        data.ID = [dictSummary objectForKey:@"id"];        ......//填入NewsData的各成员变量        [newsDataArray addObject:data];        [data release];    }    ......//其他逻辑    [[NSNotificationCenterdefaultCenter]postNotificationName:@"GetNewsListDataDone"object:newsDataArray]; }


猜测调试过程如下:

1)开始猜测问题是EGORefreshTableHeaderView大幅拖动导致的,于是把ASIHttpRequest请求数据注释掉,再次大幅拖动,程序没崩。

2)那问题一定出在ASIHttpRequest请求部分,猜测会不会是请求在子线程中做的,导致的问题,调试一下,发现请求还是在主线程中,所以也排除了这种情况。

3)一开始一直以为是ASIHttpRequest出的问题,所以精力一直放在他上面。但是后来调了1个小时,查了ASIHttpRequest的使用说明,也没查到什么疑点。

4)实在是没办法了,祭出屠龙宝刀:开始代码分段注释,运行看结果。问题出在

- (void)startGetNewsListData函数,所以从这边开始:注释if (nil != newsDataArray && newsDataArray.count > 0)    {        [newsDataArray removeAllObjects];    }

这段,运行,结果正常了,http请求也能收到返回的结果了,天哪,要是早点采用这种方法,也就不用之前尝试的1个小时了。开始分析为什么,newsDataArray前面说了,这个数据是会通过

[[NSNotificationCenter defaultCenter] postNotificationName:@"GetNewsListDataDone" object:newsDataArray]; 发送,给UITableView中使用,看接收称处的代码:......[[NSNotificationCenterdefaultCenter]addObserver:selfselector:@selector(onGetNewsListDataSuccess:)name:@"GetNewsListDataDone"object:nil];......- (void)onGetNewsListDataSuccess:(NSNotification*)notify{    NSMutableArray *receiveArray = [notify object];    ......NSInteger nNewsCount = receiveArray.count;if (nNewsCount >0)    {       m_arrNews = receiveArray;//可以看到下面使用这个成员变量m_arrNews}    [m_TableView reloadData];    ......}- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{    static NSString* strCellIdentifier =@"NewsCellIdentifier";    NewsListCell* cell = (NewsListCell *)[tableViewdequeueReusableCellWithIdentifier:strCellIdentifier];    if (cell == nil)    {       ......    }    NSInteger row = indexPath.row;    if (m_arrNews)    {//可以看到UITableView中的数据就是使用的m_arrNews        NewsData* data = (NewsData *)[m_arrNewsobjectAtIndex:row];        cell.Title = data.Title;        ......    }    ......    return cell;}

看完这些,我也知道问题出在什么地方了,我们知道,在UITableView中,在用户拖动cell,有cell 的indexPath发生变化时,就会触发这个函数:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)这个函数中我们用到了m_arrNews,而m_arrNews又是通过NSMutableArray *receiveArray = [notify object];    ......   m_arrNews = receiveArray;


这么来的,这里面全是使用的指针拷贝,原来问题就是这个:浅拷贝,先来解释一下深拷贝和浅拷贝:

   (1)深拷贝,就是新拷贝一块内存交给对象使用。会拷贝整个数据到新的地址,老的拷贝源改变和目标地址的数据就无关了。

   (2)浅拷贝,就是觉得拷贝内存太浪费,直接给你我的地址吧。当然这个地址和拷贝源相同,只要拷贝源发生改变,这个目标地址中的数据也会变化。

问题就明显了[newsDataArray removeAllObjects];导致了UITableView中的数据源发生变化,而大幅度拉动,导致了UITableView中数据刷新,进入

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *),数据失效,导致崩溃,那解决问题也很简单,使用深拷贝就可以了。- (void)onGetNewsListDataSuccess:(NSNotification*)notify{    //NSMutableArray *receiveArray = [notify object];改成NSMutableArray *receiveArray = [[NSArrayalloc]initWithArray:[notifyobject]];    ......NSInteger nNewsCount = receiveArray.count;if (nNewsCount >0)    {       m_arrNews = receiveArray;//可以看到下面使用这个成员变量m_arrNews}    [m_TableViewreloadData];    ......}

问题解决,收工。

原创粉丝点击