10

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

  • 性能:据报道,我发现的通过电子邮件地址(或电话号码)查找地址簿记录的食谱相当慢。这样做的原因是必须遍历每个地址簿记录,并且对于每个具有图像的记录,遍历所有电子邮件地址以找到匹配项。当然,对于大型地址簿来说,这可能会很耗时。

  • 表格单元格:所以我想我会收集所有需要查找图像的电子邮件地址,然后一次性找到它们。这样,我只为所有地址遍历这本书一次。但这不适用于表格单元格,其中每个单元格对应一个电子邮件地址。我要么必须在显示任何单元格之前收集所有图像(可能很慢),要么让每个单元格在加载时查找每个图像(甚至更慢,因为我需要遍历这本书以找到每个单元格的匹配项电子邮件地址)。

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

  • 异步表格单元格查找:理想情况下,我会使用诸如markjnet 的异步表格单元格更新之类的东西,以便在下载后使用图像更新表格单元格。但要使其正常工作,我必须在NSInvocationOperation每个单元格显示时为每个单元格拆分一个作业,并且如果沙箱中缺少缓存的图标。但是,我们又回到了低效地遍历每个地址簿的状态——如果您刚刚下载了一大堆新的电子邮件地址,那么这些地址簿可能会很多。

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

如果是这样,在沙箱中保存缩略图图像是缓存的最佳方法吗?如果我想创建一个新工作来更新所有缩略图,例如每天一次通讯簿中的任何更改,那么最好的方法是什么?

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

4

4 回答 4

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 条记录)就会很快。抓取图像后,只需将其缓存在字典中,这样您以后就不必重做对图像的请求。

于 2010-02-14T16:12:51.780 回答
2

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

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

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

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

于 2010-02-15T15:22:13.090 回答
1

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

于 2010-02-14T20:14:53.597 回答
0

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

于 2011-01-13T17:40:55.200 回答