4

我正在开发一个提要阅读器,我通过使用 nsxmlparser 解析 rss 提要来做到这一点。我还有从 CDATA 块中获取的缩略图对象。

-(void)parser:(NSXMLParser *)parser foundCDATA:(NSData *)CDATABlock
{
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

    NSString *someString = [[NSString alloc] initWithData:CDATABlock encoding:NSUTF8StringEncoding];
    NSString *storyImageURL = [self getFirstImageUrl:someString];

    NSURL *tempURL = [NSURL URLWithString:storyImageURL];

    NSData *tempData = [NSData dataWithContentsOfURL:tempURL];
    thumbnail = [UIImage imageWithData:tempData];

    });
}

我正在使用归档和编码和解码方法来保存图像和其他 tableview 对象。

问题是当我按原样使用上述代码时,它不会保留图像,而其他 tableview 对象(如标题和发布日期)会保留。但是当我使用上面没有 dispatch_async 的代码时,它会开始保存图像但会锁定用户界面。

如何在不锁定用户界面的情况下保留图像?

请不要使用一些库来存储和缓存图像或 Apple 的延迟加载示例。谢谢

更新:

根据提供的答案,并在观看了斯坦福大学在 iTunes 中关于多线程的讲座后,我实现了上述代码,如下所示:

  NSString *someString = [[NSString alloc] initWithData:CDATABlock encoding:NSUTF8StringEncoding];
    NSString *storyImageURL = [self getFirstImageUrl:someString];
    NSURL *tempURL = [NSURL URLWithString:storyImageURL];

    dispatch_queue_t downloadQueue = dispatch_queue_create("image downloader", NULL);

    dispatch_async(downloadQueue, ^{
        NSData *tempData = [NSData dataWithContentsOfURL:tempURL];

        dispatch_async(dispatch_get_main_queue(), ^{
            thumbnail = [UIImage imageWithData:tempData];
        });
    });
}

从逻辑上讲,现在一切都是正确的,但仍然无法正常工作。我都被阻止了如何解决这个问题。

更新:

我正在使用模型视图控制器存储方法,并且我的存储和连接代码在单独的文件中定义。下面是我使用 NSKeyedArchiver 的代码。

- (FeedChannel *) FeedFetcherWithCompletion:(void (^)(FeedChannel *, NSError *))block {

    NSURL *url = [NSURL URLWithString:@"http://techcrunch.com/feed/"];
    NSURLRequest *req = [NSURLRequest requestWithURL:url];

    FeedChannel *channel = [[FeedChannel alloc] init];

    TheConnection *connection = [[TheConnection alloc] initWithRequest:req];

    NSString *pathOfCache = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) objectAtIndex:0];
    pathOfCache = [pathOfCache stringByAppendingPathComponent:@"check.archive"];

    FeedChannel *channelCache = [NSKeyedUnarchiver unarchiveObjectWithFile:pathOfCache];


    if (!channelCachel) {
        channelCache = [[FeedChannel alloc] init];
    }
    FeedChannel *channelCopy = [channelCache copy];

    [connection setCompletionBlock:^(FeedChannel *obj, NSError *err) {
        if (!err) {
            [channelCopy addItemsFromChannel:obj];
            [NSKeyedArchiver archiveRootObject:channelCopy toFile:pathOfCache];
        }
        block(channelCopy, err);
    }];

    [connection setXmlObject:channel];
    [connection start];

    return channelCache;
}

以下是我的编码和解码方法。

- (void)encodeWithCoder:(NSCoder *)aCoder
{
    [aCoder encodeObject:title forKey:@"title"];
    [aCoder encodeObject:link forKey:@"link"];
    [aCoder encodeObject:creator forKey:@"creator"];
    [aCoder encodeObject:pubDate forKey:@"pubDate"];
    [aCoder encodeObject:thumbnail forKey:@"thumbnail"];
    //[aCoder encodeObject:UIImagePNGRepresentation(thumbnail) forKey:@"thumbnail"];
    [aCoder encodeObject:publicationDate forKey:@"publicationDate"];

}

- (id)initWithCoder:(NSCoder *)aDecoder
{
    self = [super init];
    if (self)
    {
        [self setTitle:[aDecoder decodeObjectForKey:@"title"]];
        [self setLink:[aDecoder decodeObjectForKey:@"link"]];
        [self setCreator:[aDecoder decodeObjectForKey:@"creator"]];
        [self setPubDate:[aDecoder decodeObjectForKey:@"pubDate"]];
        [self setThumbnail:[aDecoder decodeObjectForKey:@"thumbnail"]];
        //[self setThumbnail:[UIImage imageWithData:[aDecoder decodeObjectForKey:@"thumbnail"]]];
        [self setPublicationDate:[aDecoder decodeObjectForKey:@"publicationDate"]];

    }
    return self;
}

根据答案,我不知道是否必须使用一些额外的代码来保留图像。因为标题、创建者等其他对象正在准确地持续存在。

4

1 回答 1

5

阅读您的评论后,我倾向于认为问题与您如何保持相对于 dispatch_async 块的图像有关。我将尝试解释,但这只是一个猜测,因为我不知道您将图像保存在哪里:

  1. 在某些时候你parser:foundCDATA:被执行并且你异步检索你的图像;

  2. 图像将在一段时间内不存在(直到同步块完成),因此thumbnailivar/property 在完成后的一段时间内将没有正确的值parser:foundCDATA:

  3. 现在,如果您thumbnail在异步块完成之前坚持,您将坚持错误的(nil?)值;

  4. 如果您不异步执行图像的获取,那么一切正常,这一事实使我认为您过早地保留图像(在上述意义上)。

对此的修复将是仅在异步块完成后保留图像。这可能意味着:

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

  NSString *someString = [[NSString alloc] initWithData:CDATABlock encoding:NSUTF8StringEncoding];
  NSString *storyImageURL = [self getFirstImageUrl:someString];

  NSURL *tempURL = [NSURL URLWithString:storyImageURL];

  NSData *tempData = [NSData dataWithContentsOfURL:tempURL];
  thumbnail = [UIImage imageWithData:tempData];

  <code here to make the thumbnail be persisted>

});

<code here to make the thumbnail be persisted>可以是方法调用、通知帖子,任何适合您当前设计的内容。如果您的持久性代码不是线程安全的,您可能还需要使用将调用分派到主线程的技巧:

  dispatch_async(dispatch_get_main_queue(), ^{
      <persist thumbnail>;
  }

希望能帮助到你。

编辑:

经过我们的交谈,上述分析得到证实。事实上,你不是简单地下载一个图像:你是在下载一堆图像,它们都属于同一个“频道”。这意味着应用于整个通道的持久化操作只有在所有这些图像都已下载后才能执行。

I have thus forked your gist and added some logics enabling the use of GCD dispatch_group to keep track of all the download operations. You can find it here.

于 2013-01-17T21:45:09.953 回答