我正在尝试将favicons合并到UITableView中。该表基本上取了网站,我想在右边显示favicon。我最初在右边放置一个占位符图标,让后台运行一个函数。此函数获取网站的URL,解析它并尝试查找favicon。如果找不到它,它会保持相同的占位符图像;否则,它将替换为网站的favicon。我最初尝试使用[tableView reloadData],它在模拟器中工作得很好,但它确实做了奇怪和不可靠的事情(例如,它会创建一些单元格,但随后留下一个巨大的空白单元格)。无论如何,我偶然发现了reloadRowsAtIndexPaths,它似乎是我需要使用的功能。但是,结果仍然非常不可靠。我的提取功能在后台运行:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
...//other code here for showing website labels
//obtain the favicon.ico
if(!this_article.iconLoaded){
this_article.iconLoaded = TRUE;
NSDictionary *args = [NSDictionary dictionaryWithObjectsAndKeys:this_article, @"article", indexPath, @"indexPath", nil];
[self performSelectorInBackground:@selector(fetchFaviconWrapper:) withObject:args];
}
cell.favicon.image = this_article.icon;
return cell;
}
在FetchFaviconWrapper中:
- (void)fetchFaviconWrapper:(NSDictionary *)args {
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
[self fetchFavicon:[args objectForKey:@"article"]];
NSArray *paths = [NSArray arrayWithObject:[args objectForKey:@"indexPath"]];
[articleTable beginUpdates];
[articleTable reloadRowsAtIndexPaths:paths withRowAnimation:UITableViewRowAnimationFade];
[articleTable endUpdates];
[pool release];
}
基本上,fetchFavicon需要一个网站,获取主机URL,附加“/favicon.ico”,将其构建到NSData对象中,然后找到图像(如果存在)。然而,这也是非常不可靠的。在后台运行线程时替换占位符图像的最佳方法是什么?我可以在主线程上做所有事情,但这只会让表格加载缓慢。似乎有一些我忽略的东西,或者我忘记添加的东西......只是无法弄明白。
答案 0 :(得分:2)
您是否从单独的线程访问和修改UI并不完全清楚。
如果是这样,这就是您不可靠的原因。只能从主线程访问UIKit。如果您有兴趣,您会在S.O.上找到很多问题。网上有很多讨论。
如果你想保留第二个帖子,有一个解决方法。实际上,您可以使用以下命令向UI对象发送消息:
-performSelectorOnMainThread:withObject:waitUntilDone:
而不是直接从辅助线程发送它们。
如果此解决方法无法解决您的问题,那么我建议您重新设计您的应用,以便辅助线程只访问您的模型,而无需访问UI。修改UI的所有操作都应该在主线程上执行。如果您需要在模型准备就绪时在桌面上调用reloadData
,则可以使用performSelectorOnMainThread:withObject:waitUntilDone
执行此操作。
答案 1 :(得分:0)
我建议一些事情。
追加./favicon.ico并不总是准确的。查看向网站添加favicon以支持它们的各种方法。
就更换默认占位符而言,我建议使用NSNotificationCenter
通知主线程何时进行更改。
答案 2 :(得分:0)
我绝对重申没有来自后台线程的用户界面。
更改初始代码:
- (void)fetchFaviconWrapper:(NSDictionary *)args {
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
[self fetchFavicon:[args objectForKey:@"article"]];
NSArray *paths = [NSArray arrayWithObject:[args objectForKey:@"indexPath"]];
[articleTable beginUpdates];
[articleTable reloadRowsAtIndexPaths:paths
withRowAnimation:UITableViewRowAnimationFade];
[articleTable endUpdates];
[pool release];
}
到此,看看你是否还有问题
- (void)fetchFaviconWrapper:(NSDictionary *)args {
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
[self fetchFavicon:[args objectForKey:@"article"]];
NSArray *paths = [NSArray arrayWithObject:[args objectForKey:@"indexPath"]];
[self performSelectorOnMainThread:@selector(reloadData)
withObject:nil
waitUntilDone:YES];
[pool release];
}
在这种情况下,waitUntilDone是无关紧要的,并且首选表单是YES。
如果您正确处理细胞重用,那么当您遇到时髦的细胞行为时,怀疑的另一个问题是。你没有向我们展示那些代码,所以我们无法分辨。如果你在某个地方保留一个单元格的引用并将图像数据设置为 - 当你从桌面下面重新使用单元格时,你就会自己软管。 (在这种情况下,我不认为这是问题。)