在iPhone上进行异步图像缓存的最佳方法是什么?

时间:2010-02-13 23:11:25

标签: iphone image caching asynchronous abaddressbook

我正在创建一个iPhone应用程序,它将从Web API中提取数据,包括电子邮件地址。我想在表格单元格中显示与每个电子邮件地址关联的图像,因此我在地址簿中搜索图像,如果电子邮件地址不在书中,则返回默认值。这很有效,但我有一些担忧:

  • 效果:据我所知,通过电子邮件地址(或电话号码)查找地址簿记录的食谱rather slow。这样做的原因是必须遍历每个地址簿记录,并且对于每个具有图像的记录,迭代所有电子邮件地址以找到匹配。当然,这对于大型地址簿来说可能非常耗时。

  • 表格单元格:所以我想我会收集所有需要查找图片的电子邮件地址并立即查找。这样我只对所有地址遍历一次书。但这对表格单元格不起作用,其中每个单元格对应一个电子邮件地址。我要么必须在显示任何单元格之前收集所有图像(可能很慢),要么让每个单元格在加载时查找每个图像(甚至更慢,因为我需要遍历书籍才能找到匹配的< em>每个电子邮件地址)。

  • 异步查找:所以我想我会批量查找它们,但是异步使用NSInvocationOperation。对于在AddressBook中找到的每个图像,我会在应用程序沙箱中保存一个缩略图。然后每个单元格可以引用此文件并显示默认值(如果它不存在(因为它不在书中或尚未找到))。如果稍后在异步查找中找到图像,则下次需要显示图像时,它将突然出现。这可能适用于图像的周期性再生(例如,当地址簿中的图像已被更改时)。但是对于我的应用程序的任何给定实例,图像可能实际上不会显示一段时间。

  • 异步表单元格查找:理想情况下,我会使用像markjnet's asynchronous table cell updating这样的内容,一旦下载了图像单元,就会更新表格单元格。但为了实现这一点,我必须为显示的每个单元格分离NSInvocationOperation作业,并且如果沙箱中缺少缓存图标。但是后来我们又回到了无效地遍历每个地址簿的整个地址簿 - 如果你刚刚下载了一大堆新的电子邮件地址,那么它们可能很多。

所以我的问题是:其他人如何做到这一点?我正在摆弄Tweetie2,它看起来像是异步更新显示的表格单元格。我假设它正在为它需要的每个图像发送一个单独的HTTP请求。如果是这样,我想通过电子邮件地址搜索本地地址簿效率不高,那么这可能是最好的方法吗?只是不担心与搜索地址簿相关的性能问题?

如果是这样,将沙盒中的缩略图保存为缓存的最佳方法?如果我想创建一个新工作来更新所有缩略图,并且地址簿中的任何更改都说每天一次,那么这样做的最佳方法是什么?

你们其他人如何解决这类问题?建议将不胜感激!

4 个答案:

答案 0 :(得分:2)

我将使用您列出的最后一种方法(异步表单元格查找),但只查看正在显示的当前记录的图像。我重载UIScrollViewDelegate方法以找出用户何时停止滚动,然后才开始对当前可见单元格发出请求。

像这样的东西(这是我在网上找到的一个我现在找不到的教程略有修改,对不引用作者表示歉意):

- (void)loadContentForVisibleCells
{
    NSArray *cells = [self.table visibleCells];
    [cells retain];
    for (int i = 0; i < [cells count]; i++) 
    { 
        // Go through each cell in the array and call its loadContent method if it responds to it.
        AddressRecordTableCell *addressTableCell = (AddressRecordTableCell *)[[cells objectAtIndex: i] retain];
        [addressTableCell loadImage];
        [addressTableCell release];
        addressTableCell = nil;
    }
    [cells release];
}


- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView; 
{
    // Method is called when the decelerating comes to a stop.
    // Pass visible cells to the cell loading function. If possible change 
    // scrollView to a pointer to your table cell to avoid compiler warnings
    [self loadContentForVisibleCells]; 
}


- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate;
{
    if (!decelerate) 
    {
       [self loadContentForVisibleCells]; 
    }
}

一旦你知道当前可以看到哪些地址记录,只需搜索那些(可能是5-7条记录)就会很快。抓取图像后,只需将其缓存在字典中,这样就不必在以后重做对图像的请求。

答案 1 :(得分:2)

无论您使用什么策略进行实际的图像缓存,每次获得一批电子邮件地址时,我只会通过地址簿数据,如果可能的话。 (是的,我会异步地这样做。)

创建一个NSMutableDictionary,它将作为搜索结果的内存缓存。使用下载中的每个电子邮件地址作为密钥初始化此字典,并将标记作为该密钥的值(例如[NSNull null])。

接下来,遍历通讯簿中的每个ABRecordRef,调用ABRecordCopyValue(record, kABPersonEmailProperty)并循环返回返回的每个ABMultiValue中的结果。如果任何电子邮件地址是缓存中的密钥,请将[NSNumber numberWithInt:ABRecordGetRecordId(record)]设置为字典中该密钥的值。

使用此词典作为查找索引,您可以快速获取ABRecordRefs的图像,仅显示当前在表格视图中显示的电子邮件地址,给出用户当前的滚动位置,如hoopjones的答案所示。如果您的应用程序需要“最新版本”级别,您可以添加地址簿更改侦听器以使缓存无效,触发另一个索引操作,然后更新视图。

答案 2 :(得分:1)

您似乎尝试在UITableView中实现加载延迟图像。 Apple有一个很好的例子,我在这里引用它: Lazy load images in UITableView

答案 3 :(得分:0)

仅供参考,我发布了一个免费,功能强大且易于使用的库,用于执行异步图像加载和快速文件缓存:HJ Managed Objects http://www.markj.net/asynchronous-loading-caching-images-iphone-hjobjman/