14

我注意到像 Intagram 这样的应用程序UICollectionViews用来显示照片的提要。我还注意到这些照片的单元格在实际照片完全下载之前以某种方式放置在屏幕上。然后,当下载完成时,该照片会很好地显示在正确的单元格中。

我想复制该功能,但我不知道如何去做。

我目前正在下载一个大型JSON对象,我将其转换为NSDictionaries数组。每个都NSDictionary包含有关每张照片的信息,并且在这些信息中,会显示一个 URL。在这个网址,我可以找到我需要下载和显示的相应图片UICollectionViewCells

就目前而言,我迭代此列表并为我看到的每个 URL 启动下载。下载完成后,我使用[self.collectionView reloadData]. 但是,正如你可以想象的那样,如果我有 30 个单元格都需要一个图像,那么就会有很多reloadData调用。

我正在使用 AFNetworking 来处理下载,这是我根据之前提到的 URL 调用的方法:

-(void) downloadFeedImages:(NSString *) photoURL imageDescs:(NSDictionary*)imageDescs photoId:(NSString *)photoID{
    NSArray *paths = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES);
    NSString *directory = [paths objectAtIndex:0];

    NSString* foofile = [directory stringByAppendingPathComponent:photoID];

    if([[NSFileManager defaultManager] fileExistsAtPath:foofile]){
        // IF IMAGE IS CACHED
        [self.collectionView reloadData];
        return;
    }

    NSLog(@"photoURL: %@", photoURL);
    NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:photoURL]];
    AFImageRequestOperation *operation = [AFImageRequestOperation imageRequestOperationWithRequest:request
                                                                              imageProcessingBlock:nil
                                                                                           success:^(NSURLRequest *request, NSHTTPURLResponse *response, UIImage *image) {
                                                                                               // Save Image
                                                                                               NSLog(@"URL-RESPONSE:%@", image);
                                                                                               NSString *myFile = [directory stringByAppendingPathComponent:photoID];

                                                                                               NSData *imageData = UIImagePNGRepresentation(image);
                                                                                               [imageData writeToFile:myFile  atomically: YES];
                                                                                               [self.collectionView reloadData];
                                                                                           }
                                                                                           failure:^(NSURLRequest *request, NSHTTPURLResponse *response, NSError *error) {
                                                                                               NSLog(@"ERROR: %@", [error localizedDescription]);
                                                                                           }];
    [[WebAPI sharedInstance] enqueueHTTPRequestOperation:operation];
}

所以基本上,我想知道如何实现 Instagram 和类似应用程序在显示带有图像的提要时所具有的功能。另外,我想知道为每个单元格启动下载的好方法,当下载完成时,更新那个单元格,而不是使用[reloadData]重绘整个视图

谢谢

4

3 回答 3

16

您要实现的技术称为延迟加载。由于您正在使用AFNetworking它,因此在您的情况下实现这一点会更容易。您的每个集合视图单元格都需要有一个UIImageView来显示图像。使用UIImageView+AFNetworking.h类别并通过调用方法设置正确的图像URL

- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath
{
    // ....

    [cell.imageView setImageWithURL:imageURL placeholderImage:[UIImage imageNamed:@"placeholder.png"]];

    // ...
    return cell;
}

占位符是在下载所需图像之前将显示的图像。这将为您完成所需的工作。

注意:默认情况下,URL 请求的缓存策略为NSURLCacheStorageAllowed30 秒,超时间隔为 30 秒,并且设置为不处理 cookie。要以不同方式配置 URL 请求,请使用setImageWithURLRequest:placeholderImage:success:failure:.

另外,供您参考,如果您想自己实现图像的延迟加载,请遵循此Apple 示例代码。这是用于UITableView但同样的技术也可以用于UICollectionView

希望有帮助!

于 2013-08-27T07:38:53.370 回答
2

利用

[self.collectionView reloadItemsAtIndexPaths:[NSArray arrayWithObject:indexPath]];

代替

[self.collectionView reloadData];

indexPathNSIndexPath对应对象的UICollectionViewCell对象

于 2013-08-27T08:01:53.103 回答
1

使用以下在cellForItemAtIndexPath委托方法中的单元格中加载图像异步

let url = URL(string: productModel.productImage)

    let data = try? Data(contentsOf: url!)
    DispatchQueue.main.async(execute: {
        collectionViewCell.imageView.image = UIImage(data: data!)
    });

在 ViewController 中实现协议“UICollectionViewDataSourcePrefetching”为

类视图控制器: UIViewController , UICollectionViewDataSourcePrefetching {

在情节提要(见附图)或以编程方式将以下代表设置到您的集合视图

ViewController 的 viewDidLoad方法中

collectionView.delegate = self
collectionView.dataSource = self
collectionView.prefetchDataSource = self
于 2016-10-06T06:55:59.200 回答