我正在尝试在单元格中加载已解析的数据,但问题是它是同步发生的,UitableView在数据加载完成之前不会显示。我试图通过使用performSelectorInBackground解决问题,但现在数据没有加载到单元格中,直到我开始滚动。这是我的代码:
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
[self performSelectorInBackground:@selector(fethchData) withObject:nil];
}
- (void)viewDidUnload
{
[super viewDidUnload];
// Release any retained subviews of the main view.
self.listData = nil;
self.plot=nil;
}
-(void) fethchData
{
NSError *error = nil;
NSURL *url=[[NSURL alloc] initWithString:@"http://www.website.com/"];
NSString *strin=[[NSString alloc] initWithContentsOfURL:url encoding:NSUTF8StringEncoding error:nil];
HTMLParser *parser = [[HTMLParser alloc] initWithString:strin error:&error];
if (error) {
NSLog(@"Error: %@", error);
return;
}
listData =[[NSMutableArray alloc] init];
plot=[[NSMutableArray alloc] init];
HTMLNode *bodyNode = [parser body];
NSArray *contentNodes = [bodyNode findChildTags:@"p"];
for (HTMLNode *inputNode in contentNodes) {
[plot addObject:[[inputNode allContents] stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]]];
}
NSArray *divNodes = [bodyNode findChildTags:@"h2"];
for (HTMLNode *inputNode in divNodes) {
[listData addObject:[[inputNode allContents] stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]]];
}
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
static NSString *CellIdentifier = @"Cell";
//here you check for PreCreated cell.
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if (cell == nil) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:CellIdentifier];
cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;
}
//Fill the cells...
cell.textLabel.text = [listData objectAtIndex:indexPath.row];
cell.textLabel.font = [UIFont boldSystemFontOfSize:14];
cell.textLabel.numberOfLines=6;
cell.textLabel.textColor=[UIColor colorWithHue:0.7 saturation:1 brightness:0.4 alpha:1];
cell.detailTextLabel.text=[plot objectAtIndex:indexPath.row];
cell.detailTextLabel.font=[UIFont systemFontOfSize:11];
cell.detailTextLabel.numberOfLines=6;
return cell;
}
答案 0 :(得分:129)
在数据成功加载后将其放在 :
dispatch_async(dispatch_get_main_queue(), ^{
[self.tableView reloadData];
});
这解决了当您不在主线程中时调用GUI更新的问题。
此代码使用Apple的GCD技术强制重装数据功能在主线程上运行。了解更多 Concurrency Programming Guide以获得更多理解(这是一个相当大的字段,因此很难在评论中解释) 无论如何,如果你不太了解它并不是很好,因为它会导致程序在一些罕见的情况下崩溃。
答案 1 :(得分:14)
对于swift 3:
DispatchQueue.main.async(execute: { () -> Void in
self.tableView.reloadData()
})
对于swift 2:
dispatch_async(dispatch_get_main_queue(), { () -> Void in
self.tableView.reloadData()
})
答案 2 :(得分:9)
您真正需要做的就是每当您对后端数据进行更新时,请致电
[tableView reloadData];
由于这是同步发生的,你应该有一个像
这样的功能-(void) updateTable
{
[tableView reloadData];
}
并在下载调用中添加数据后
[self performSelectorOnMainThread:@selector(updateTable) withObject:nil waitUntilDone:NO];
答案 3 :(得分:2)
你可以使用[cell setNeedsDisplay]; 例如:
dispatch_async(dispatch_get_main_queue(), ^{
[cell setNeedsDisplay];
[cell.contentView addSubview:yourView];
});
答案 4 :(得分:1)
我遇到了完全相同的问题!我希望在视图控制器出现之前完全填充UITableView。 Envil的帖子给了我所需的信息,但我的解决方案最终变得不同了。
这就是我所做的(为了适应提问者的背景而改造)。
- (void)viewDidLoad {
[super viewDidLoad];
[self performSelectorInBackground:@selector(fethchData) withObject:nil];
}
- (void)viewWillAppear {
[tableView reloadData];
}
答案 5 :(得分:1)
我遇到了这个问题,我整天都在处理它。
我使用静态单元格并且reloadData导致错误加载,它只显示可见单元格并删除其他单元格。 我注意到的是当我向下滚动(y为负值)正确加载的单元格时,所以我编写了这段代码并且它有效,即使我不想以这种方式让它。
如果找到更好的解决方案,请拍摄。
-(void)reloadTableView{
CGPoint point = self.tableSettings.tableView.contentOffset;
[self.tableSettings.tableView reloadData];
[UIView animateWithDuration:.001f animations:^{
[self.tableSettings.tableView setContentOffset:CGPointMake(point.x, -10)];
} completion:^(BOOL finished) {
[UIView animateWithDuration:.001f animations:^{
[self.tableSettings.tableView setContentOffset:point];
} completion:^(BOOL finished) {
}];
}];
}
答案 6 :(得分:0)
首先,这半解决了我的相关问题。我想在表格单元格中舍入图像的角落。异步调度修复了一些但不是所有图像的问题。有任何想法吗?
其次,我认为你应该通过使用这样的闭包捕获列表来避免创建一个强大的引用循环:
k / (k^2 / log(k^2)) = 2 log(k) / k
答案 7 :(得分:0)
在使用自动调整大小的单元格时,我遇到了同样的问题,我发现将估计高度设置为50可以解决此问题。
在tableView本身上设置估算的高度,或从UITableViewDelegate中的estimatedHeightForRowAt
返回估算值。
只要估计值大于0,它似乎就可以工作。