UICollectionView在滚动时重新加载错误的图像

时间:2014-02-19 15:25:58

标签: ios uicollectionview afnetworking

在下面的代码中,我从文本文件下载图像URL,将其存储在NSMutableArray中,然后从CellforItemAtIndexPath中的这些URL下载图像。但是,当我向上和向下滚动片刻时,错误的图像位于错误的位置。它纠正了一秒钟,但我想摆脱滚动问题。这是代码:

- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath{
    static NSString *identifier = @"Cell2";

    UICollectionViewCell *cell = [collectionView dequeueReusableCellWithReuseIdentifier:identifier forIndexPath:indexPath];
    AsyncImageView *recipeImageView = (AsyncImageView *)[cell viewWithTag:100];
    UILabel *titleView = (UILabel *)[cell viewWithTag:101];
    titleView.numberOfLines = 3;
    UIImageView *overlay = (UIImageView *)[cell viewWithTag:111];
    FXBlurView *blurView = (FXBlurView *)[cell viewWithTag:110];
    blurView.tintColor = [UIColor colorWithRed:51.0 green:51.0 blue:51.0 alpha:1];


    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0), ^{
        hi = [self videoTitle];
        images = [self firstPhoto];
        // switch to a background thread and perform your expensive operation


        // switch back to the main thread to update your UI


        dispatch_async(dispatch_get_main_queue(), ^{
            NSString *imgSrc;

            NSString *url = images[indexPath.row];
            imgSrc = url;
            if (imgSrc != nil && [imgSrc length] != 0 ) {
                [recipeImageView setImageWithURL:[NSURL URLWithString:images[indexPath.row]] placeholderImage:[UIImage imageNamed:@"red.png"]];

            }
            else {
                NSLog(@"noimage");
                recipeImageView.image = [UIImage imageNamed:@"red.png"];
            }
            titleView.text = hi[indexPath.row];


        });
    });

    return cell;


    // recipeImageView.image = [UIImage imageNamed:[videoList objectAtIndex:indexPath.row]];

}

有什么想法吗?谢谢。

3 个答案:

答案 0 :(得分:5)

我不确定AsyncImageView是什么,所以首先让我回答你的问题,好像它是UIImageView


在回调块中(当您返回主队列时),您指的是recipeImageView。但是,如果用户在下载过程中滚动,则cell对象已被重用。

进入回调后,您必须假设所有对视图的引用(cellrecipeImageViewblurView等都指向错误的内容。您需要使用数据模型找到正确的视图。

一种常见方法是使用NSIndexPath查找单元格:

UICollectionViewCell *correctCell = [collectionView cellForItemAtIndexPath:indexPath]:
UIImageView *correctRecipeImageView = (UIImageView *)[correctCell viewWithTag:100];
/* Update your image now… */

请注意,这仅在始终保证一个indexPath指向相同内容时才有效 - 如果用户可以插入/删除行,那么您还需要找出正确的indexPath,因为它可能也有所改变:

NSIndexPath *correctIndexPath  = /* Use your data model to find the correct index path */
UICollectionViewCell *correctCell = [collectionView cellForItemAtIndexPath:correctIndexPath]:
UIImageView *correctRecipeImageView = (UIImageView *)[correctCell viewWithTag:100];
/* Update your image now… */

那就是说,如果AsyncImageView是其他一些应该为你处理所有这些的类,那么你可以在主线程上调用setImageWithURL: placeholderImage:,它应该处理所有的异步加载代表你。如果没有,我想推荐SDWebImage库,它可以:https://github.com/rs/SDWebImage

答案 1 :(得分:3)

我注意到一堆没有评论的downvotes,在再次阅读我的答案后,我猜原来接受的答案(在线下)可能会更好。

这些问题还有一些问题。 UITableView只创建足够的UITableViewCells来显示视图中的每个单元格以及将滚动到视图中的内容。其他单元格在滚出视图后将重新使用。

这意味着当您快速滚动时,单元格会多次设置以获取图像异步,然后显示它,因为获取图像非常慢。图像加载后,它将显示在最初调用它的单元格上,但该单元格可能已经重新使用。

现在我有一个bufferedImage:UIImage?我用作UITableViewCells数据的任何数据类的属性。这最初总是为零,所以我获取图像异步,当dispatch_async返回时,它会将它存储在那里,所以下次我需要显示这个单元格它已经准备就绪,如果它仍然是同一个单元格,我也会在单元格中显示它

所以正确的做法是:

  • 开始使用您的数据对象配置单元格
  • 将数据对象存储在您的单元格中,以便稍后再参考
  • 检查数据对象是否已在bufferedImage中设置了图像,如果是,则显示它并且就是这样
  • 如果还没有图像,请将当前图像重置为空或占位符,然后开始在后台下载图像
  • 异步调用返回时将图像存储在bufferedImage属性
  • 检查当前单元格中存储的对象是否仍然是您为该图像获取的对象(这非常重要),如果不是,则执行任何操作
  • 如果您仍然在同一个单元格中显示图片

因此,您在异步调用中传递了一个对象,以便存储所获取的图像。还有另一个对象可以在当前单元格中进行比较。通常,在获取图像之前,单元格已经被重用,因此在下载图像时这些对象会有所不同。


滚动和旋转CollectionViews总是有点问题。我总是有同样的问题,图像出现在错误的单元格中。对于集合视图,会进行某种优化,这会影响图像,但不会影响标签。对我来说这是一个错误,但在三个iOS发布后它仍然没有解决。

您需要实施以下内容:

https://developer.apple.com/library/ios/documentation/uikit/reference/UICollectionViewLayout_class/Reference/Reference.html#//apple_ref/occ/instm/UICollectionViewLayout/shouldInvalidateLayoutForBoundsChange

这应该总是返回YES基本上总是刷新它。如果这可以解决您的问题,那么您可能会考虑不执行整个重绘,因为它对性能有害。我个人花了很多时间来计算一个屏幕已经通过或线路已经过去等等,但它变得很吵,所以我最后只回到YES。 YMMV。

答案 2 :(得分:0)

我看到您已经回答了这个问题,但仍然想分享我的经验。

我遇到了同样的问题。

在我看来,此问题的原因是以下功能: collectionView.dequeueReusableCell (...)。

每种情况下,单元格的类型都是相同的,但是数据是不同的:

  • 在某些情况下,特定单元格的数据包含图像。
  • 在某些情况下,特定单元格的数据不包含图像,为空。

当图像为零时,我没有将图像设置为imageView。在这种情况下,集合视图会重用其他单元格的图像(当前单元格的图像错误)。

解决方案非常简单:如果imageView不包含图像,只需设置nil即可。

希望这会对您有所帮助。