1

简要介绍一下我在做什么:我有两个数组,它们都包含 50% 的表格视图信息。为什么?因为其中一个阵列从互联网上提取当前信息,而另一个阵列保存了用户数据。我不知道如何以一种不混乱的方式从互联网上获取当前信息,因为我是 Objective-C 的业余爱好者,更不用说在 Objective-C 中建立网络了。无论如何,因此互联网数组正在使用 AFNetworking 提取与已保存数组(保存在核心数据中)中的对象相对应的信息。因此它是我想要的异步的。现在问题来了,或者至少我可以从这种情况中得到什么。

我正在做一个 for 循环,它有效地计算保存数组中的对象并传递每个对象的唯一 ID,以便可以下载、解析来自 Internet 的相应信息并将其添加到 Internet 数组中。但是,由于网络是异步的,因此循环实际上是一次从 Internet 下载所有信息。因此,对象将按照先下载的顺序写入互联网阵列。因此 savedArray[0] 不对应于 internetArray[0] 中的对象。正如您可以想象的那样,这是一个相当大的缺陷,因为当我将值插入到我的 tableView 中时,没有什么真正匹配/有意义。

我真的在寻找一种方法来推迟信息的下载,直到之前的下载完成并添加到 internetArray,我到底该怎么做?现在是代码。这是我获得适当密钥的地方:

for ( int i = 0; i < [mySavedObjects count]; i++) {
    MySavedObject* mySavedObject = [mySavedObjects objectAtIndex:i];
    [self retrieveMissingInformation: myObject.Id];
    [self.tableView reloadData];
}

这是我实际获取信息的地方(为了空间而简化):

- (void)retrieveMissingInformation:(NSString *) Id
{
 // create URL here.
AFJSONRequestOperation *operation = [AFJSONRequestOperation JSONRequestOperationWithRequest:request
              success:^(NSURLRequest *request, NSHTTPURLResponse *response, id JSON) {
                        NSLog(@"Success");
                        // Do some safety checks & parse the JSON, returning a searchResult.
                        [searchResults addObject:searchResult];
            } failure:^(NSURLRequest *request, NSHTTPURLResponse *response, NSError *error, id JSON)
                        // Do some error messages etc.
}];

[queue addOperation:operation]; //where queue is an ivar NSOperationQueue.
}

最后,在 cellForRowAtIndexpath 中,我使用了这两个:

MySavedObject *mySavedObject = [mySavedObjects objectAtIndex:indexPath.row];

&

SearchResult *searchResult = [searchResults objectAtIndex:indexPath.row];

获取单元格的值。

对不起,巨大的文字墙。我真的不够好,无法有效地解释事情,经常在术语上结结巴巴,不得不求助于自己的例子。订购这个烂摊子的任何帮助,我将不胜感激。

问候,
迈克。

4

3 回答 3

2

好的工作伙伴,你偶然发现了一个常见的设计模式。

正如您所看到的,对于异步请求,执行几乎是同时发生的,因此当您从 for 循环的角度考虑“队列”时,您无法保证顺序。

您需要考虑下载队列的方式是递归。

换句话说,而不是:

// your current method of queue-ing the submission of data
for(int i = 0; i < arrData.count; i++)
{
    [self doAsyncDataSendForIndex:i];
}

你需要开始做这样的事情:

// perform the first submit for the first element in your array
[self doAsyncDataSendForIndex:dataIndex];


// a completion callback method
-(void)asyncFinishDownloading
{
    [self parseResult:serverReturnedData];
}

-(void)parseResult:(id)serverData
{
    // do all the processing of the data that you need here

    ...

    // update index for next round
    dataIndex++;

    // -----------------------------------------------------------------
    // now at the end of this method, perform a recursive call 
    // of your async download method again 
    // -----------------------------------------------------------------
    [self doAsyncDataSendForIndex:dataIndex];
}

使用对下载进行排队的回调委托方式,您是在告诉代码仅在处理完最后一次异步下载后才下载下一个数据。

现在您可以使用许多库来帮助您处理异步内容。我自己使用 ASIHttpRequest 和 ASIFormData,你也可以使用更新的 AFNetworking。

-(void)asyncSubmitData:(id)someData
{
    NSURL *url = [NSURL urlWithString:@"http://www.yourserver.com/...."];

    ASIFormDataRequest *request = [ASIFormDataRequest requestWithURL:url];

    // setup your POST request parameters
    [request setPostValue:someData forKey:@"someKey"];
    ...

    // setup the completion callback delegate action
    [request setCompletionBlock:^{
        // process your server returned data here
        NSString *serverResponse = [request responseString]; // raw server response

        ...

        // increment your index and call your download method again
        dataIndex++;

        [self asyncSubmitData:[myArray objectAtIndex:dataIndex]];
    }];


    // setup fail block in case the server fails
    [request setFailedBlock:^{
        NSLog(@"Server error: %@", [[request error] localizedDescription];
    }];


    // start the asynchronous request now
    [request startAsynchronous];
}

希望有帮助。

于 2012-11-30T02:11:51.190 回答
1

推迟信息的下载并有效地使所有网络请求串行(一次一个)会起作用,但这不是一个很好的解决方案,因为进行并发网络请求通常很方便 - 例如,允许四个请求一次可能比连续发出所有请求花费的时间更少。

一个更好的解决方案是在返回时正确处理一条数据。为此,您需要以更结构化的方式处理返回的数据。不只是将其附加到数组中,请执行以下操作之一:

A.将返回的数据放入字典中。字典可以从索引或适当的键映射到返回的数据。(提示:如果你想将一个 int 作为键放入字典中,请构造一个包含该 int 的 NSNumber。) 建议使用此选项而不是 B。

B.将返回的数据插入数组。为了使这些有意义,在您进行任何获取之前,您应该用表示“尚未获取此数据位”的占位符对象填充您的数组。一种可能的占位符对象选择是 NSNull - 它只是一个表示 null/nil 的对象。

因此,您可以在 cellForIndexPath: 中返回正确的项目,方法是浸入您的字典或数组。

这两个数据结构的想法只是我想到的最明显的想法,毫无疑问还有其他方法可以解决这个问题。

于 2012-11-30T00:21:58.263 回答
1

我想我理解你的问题,听起来你使用了错误的数据结构来存储数据。

您有一个正在生成请求的数组,这些请求的结果存储在一个单独的、不相关的数组中。

当您使用 objectId 生成信息请求时,您能否将结果存储在 NSDictionary 而不是数组中,并使用 objectId 作为该数组的键?

您在成功方法中有 NSURLRequest,因此如果它是查询字符串上的参数,您应该能够通过 NSURL.query 从请求中检索您的 objectId

于 2012-11-30T00:16:08.933 回答