1

我正在使用CMIS(Content management interoperability services)从露天服务器下载数据。我正在使用以下代码,它在某种程度上可以正常工作,但是当应用程序进入后台时,网络连接丢失,当应用程序进入前台时,它会尝试重试下载并失败说连接错误。由于我是新手,任何帮助将不胜感激。

- (void)testFileDownload
{
    [self runTest:^
     {
         [self.session retrieveObjectByPath:@"/ios-test" completionBlock:^(CMISObject *object, NSError *error) {
             CMISFolder *testFolder = (CMISFolder *)object;
             STAssertNil(error, @"Error while retrieving folder: %@", [error description]);
             STAssertNotNil(testFolder, @"folder object should not be nil");

             CMISOperationContext *operationContext = [CMISOperationContext defaultOperationContext];
             operationContext.maxItemsPerPage = 100;
             [testFolder retrieveChildrenWithOperationContext:operationContext completionBlock:^(CMISPagedResult *childrenResult, NSError *error) {
                 STAssertNil(error, @"Got error while retrieving children: %@", [error description]);
                 STAssertNotNil(childrenResult, @"childrenCollection should not be nil");

                 NSArray *children = childrenResult.resultArray;
                 STAssertNotNil(children, @"children should not be nil");
                 STAssertTrue([children count] >= 3, @"There should be at least 3 children");

                 CMISDocument *randomDoc = nil;
                 for (CMISObject *object in children)
                 {
                     if ([object class] == [CMISDocument class])
                     {
                         randomDoc = (CMISDocument *)object;
                     }
                 }

                 STAssertNotNil(randomDoc, @"Can only continue test if test folder contains at least one document");
                 NSLog(@"Fetching content stream for document %@", randomDoc.name);

                 // Writing content of CMIS document to local file
                 NSString *filePath = [NSString stringWithFormat:@"%@/testfile", NSTemporaryDirectory()];
                 //                NSString *filePath = @"testfile";
                 [randomDoc downloadContentToFile:filePath
                                  completionBlock:^(NSError *error) {
                                      if (error == nil) {
                                          // Assert File exists and check file length
                                          STAssertTrue([[NSFileManager defaultManager] fileExistsAtPath:filePath], @"File does not exist");
                                          NSError *fileError = nil;
                                          NSDictionary *fileAttributes = [[NSFileManager defaultManager] attributesOfItemAtPath:filePath error:&fileError];
                                          STAssertNil(fileError, @"Could not verify attributes of file %@: %@", filePath, [fileError description]);
                                          STAssertTrue([fileAttributes fileSize] > 10, @"Expected a file of at least 10 bytes, but found one of %d bytes", [fileAttributes fileSize]);

                                          // Nice boys clean up after themselves
                                          [[NSFileManager defaultManager] removeItemAtPath:filePath error:&fileError];
                                          STAssertNil(fileError, @"Could not remove file %@: %@", filePath, [fileError description]);
                                      } else {
                                          STAssertNil(error, @"Error while writing content: %@", [error description]);
                                      }
                                      self.testCompleted = YES;
                                  } progressBlock:nil];
             }];
         }];
     }];
}

当用户按下 home 键时不会发生连接失败。仅当磁盖关闭或超时时才会失败。

4

2 回答 2

2

当应用程序移至后台时,操作系统会在应用程序暂停之前为应用程序提供 5 秒的时间来完成其正在执行的操作(保留 RAM,但停止应用程序接收任何消息或执行任何操作)。如果您有一个任务需要在用户按下主页按钮时运行完成,您可以使用后台任务。从苹果的文档中:

您的应用委托的 applicationDidEnterBackground: 方法大约有 5 秒的时间来完成所有任务并返回。在实践中,此方法应尽快返回。如果该方法在时间用完之前没有返回,您的应用程序将被终止并从内存中清除。如果您仍然需要更多时间来执行任务,请调用 beginBackgroundTaskWithExpirationHandler: 方法来请求后台执行时间,然后在辅助线程中启动任何长时间运行的任务。无论您是否启动任何后台任务,applicationDidEnterBackground: 方法仍必须在 5 秒内退出。

注意:还会发送 UIApplicationDidEnterBackgroundNotification 通知,让您的应用程序中感兴趣的部分知道它正在进入后台。您应用中的对象可以使用默认通知中心来注册此通知。

来自http://developer.apple.com/library/ios/#documentation/iphone/conceptual/iphoneosprogrammingguide/ManagingYourApplicationsFlow/ManagingYourApplicationsFlow.html

于 2013-05-15T18:34:52.260 回答
1

使用可达性代码 下载后尝试使用此代码保存数据:

-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data 
{
//try to access that local file for writing to it...
NSFileHandle *hFile = [NSFileHandle fileHandleForWritingAtPath:self.localPath];
//did we succeed in opening the existing file?
if (!hFile) 
{   //nope->create that file!
   [[NSFileManager defaultManager] createFileAtPath:self.localPath contents:nil attributes:nil];
   //try to open it again...
   hFile = [NSFileHandle fileHandleForWritingAtPath:self.localPath];
}
//did we finally get an accessable file?
if (!hFile)
{   //nope->bomb out!
   NSLog("could not write to file %@", self.localPath); 
   return;
}
//we never know - hence we better catch possible exceptions!
@try 
{
   //seek to the end of the file
   [hFile seekToEndOfFile];
   //finally write our data to it
   [hFile writeData:data];
}
@catch (NSException * e) 
{
   NSLog("exception when writing to file %@", self.localPath); 
   result = NO;
}
[hFile closeFile];
}
于 2013-05-15T19:05:59.407 回答