GCD UITableView asynchronous load images, wrong cells are loaded until new image download

前端 未结 3 1048
挽巷
挽巷 2020-12-09 12:55

I have a UITableView with custom cells. I load images asynchronously using Grand Central Dispatch. Everything works fine, but when I scroll down, previously loaded images ar

3条回答
  •  春和景丽
    2020-12-09 13:29

    Rather than capturing the cell you need to capture the index path, then get the cell back using:

    UITableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];
    

    That way, if the cell is now off screen you'll get nil back and the image won't be set on the wrong cell.

    The other thing you need to add after your dispatch_async() is a cell.imageView.image=somePlaceholderImage.

    E.g.:

    if (![[NSFileManager defaultManager] fileExistsAtPath:[path stringByAppendingPathComponent:@"image.png"]])
    {
        dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH,  0ul);
        dispatch_async(queue, ^{
            NSString *url=[pat stringByAppendingPathComponent:@"comments.txt"];
            NSString *u=[NSString stringWithContentsOfFile:url encoding:NSUTF8StringEncoding error:nil];
            NSURL *imageURL=[NSURL URLWithString:u];
            NSData *image=[NSData dataWithContentsOfURL:imageURL];
            [image writeToFile:[pat stringByAppendingPathComponent:@"image.png"] atomically:YES];
            dispatch_sync(dispatch_get_main_queue(), ^{
                UITableViewCell *cell = [tableView cellForRowAtIndexPath:indexPath];
                cell.imageView.image=[UIImage imageWithContentsOfFile:[pat stringByAppendingPathComponent:@"image.png"]];
                [cell setNeedsLayout];
                NSLog(@"Download");
            });
        });
        cell.imageView.image=[UIImage imageNamed:@"placeholder"];
    }
    else
    {
        NSLog(@"cache");
        cell.imageView.image=[UIImage imageWithContentsOfFile:[pat stringByAppendingPathComponent:@"image.png"]];
    }
    

提交回复
热议问题