2016-09-24 34 views
0

我需要同時打上百個請求才能上傳文件。我正在使用AFNetworking,我只是從for循環中調用一個方法。但它在一些文件中給我內部服務器錯誤。服務器端沒有任何東西。該過程從Web成功執行。使用AFNetworking同時打出數百個請求

這裏是我的代碼:

for (UIImage *img in imgages) { 
     [self uploadImage:img]; 
     } 


-(void) uploadImage:(UIImage *)image 
{ 



    // NSDictionary *[email protected]{x:x } 
    NSData *jsonData = [NSJSONSerialization dataWithJSONObject:someData 
                 options:0 

                 error:nil]; 

     NSMutableURLRequest *request = [[AFHTTPRequestSerializer serializer] multipartFormRequestWithMethod:@"POST" URLString:[BASE_URL stringByAppendingString:@"xyz.com"] parameters:nil constructingBodyWithBlock:^(id<AFMultipartFormData> formData) { 
     [formData appendPartWithFormData:jsonData name:@"abc"]; 

     if (imgData) { 

      [formData appendPartWithFormData:imgData name:@"file"]; 

     } 
    } error:nil]; 

    AFURLSessionManager *manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]]; 

    NSURLSessionUploadTask *uploadTask; 

    manager.responseSerializer = [AFHTTPResponseSerializer serializer]; 

    uploadTask = [manager 
        uploadTaskWithStreamedRequest:request 
        progress:nil completionHandler:^(NSURLResponse * _Nonnull response, id _Nullable responseObject, NSError * _Nullable error) { 

        }]; 

    [uploadTask resume]; 

} 

什麼是實現上述功能的最佳方式?

+1

您收到的具體錯誤是什麼? iOS具有每個主機的併發請求限制,但如果這是您遇到的情況,您將會從超出的請求中獲取超時。 – Avi

+1

嘗試使用'dispatch_apply'而不是fo-loop – Enix

+0

@Vikas請接受答案,如果它的工作 –

回答

1

嘗試使用下面的代碼來同時上傳文件。

-(id)init { 
self = [super init]; 
if(self) { 
    backgroundQueue = dispatch_queue_create("com.bgqueue", NULL); 

    dispatch_once(&onceToken, ^{ 

     NSURLSessionConfiguration *sessionConfiguration = [NSURLSessionConfiguration backgroundSessionConfigurationWithIdentifier:@"com.Demo-Upload"]; 
     sessionConfiguration.HTTPMaximumConnectionsPerHost = 10; 
     manager = [[AFURLSessionManager alloc] initWithSessionConfiguration:sessionConfiguration]; 
    }); 
} 
return self; 
} 

- (void)processFiles{ 
    for (UIImage *img in imgages) { 
     dispatch_async(backgroundQueue, ^(void) { 
      [self uploadImage:img]; 
     }); 
    } 
} 


-(void) uploadImage:(UIImage *)image 
{ 
    // Prepare a temporary file to store the multipart request prior to sending it to the server due to an alleged 
    // bug in NSURLSessionTask. 
    NSString* tmpFilename = [NSString stringWithFormat:@"%f", [NSDate timeIntervalSinceReferenceDate]]; 
    NSURL* tmpFileUrl = [NSURL fileURLWithPath:[NSTemporaryDirectory() stringByAppendingPathComponent:tmpFilename]]; 

    // Create a multipart form request. 
    NSMutableURLRequest *request = [[AFHTTPRequestSerializer serializer] multipartFormRequestWithMethod:@"POST" URLString:[BASE_URL stringByAppendingString:@"xyz.com"] parameters:nil constructingBodyWithBlock:^(id<AFMultipartFormData> formData) { 
    [formData appendPartWithFormData:jsonData name:@"abc"]; 

     if (imgData) { 
      [formData appendPartWithFormData:imgData name:@"file"]; 
     } 
    } error:nil]; 

    // Dump multipart request into the temporary file. 
    [[AFHTTPRequestSerializer serializer] requestWithMultipartFormRequest:request 
              writingStreamContentsToFile:tmpFileUrl 
                completionHandler:^(NSError *error) 
    { 
      // Once the multipart form is serialized into a temporary file, we can initialize 
      // the actual HTTP request using session manager. 


      // Here note that we are submitting the initial multipart request. We are, however, 
      // forcing the body stream to be read from the temporary file. 

      self.uploadTask = [manager uploadTaskWithRequest:request 
           fromFile:tmpFileUrl 
           progress:^(NSProgress * _Nonnull uploadProgress) 
              { 
               NSLog(@"Progress… %f", uploadProgress.fractionCompleted); 
              } 
           completionHandler:^(NSURLResponse *response, id responseObject, NSError *error) 
              { 
               // Cleanup: remove temporary file. 
               [[NSFileManager defaultManager] removeItemAtURL:tmpFileUrl error:nil]; 

               // Do something with the result. 
               if (error) 
               { 
               //Print Error 
               } else 
               { 
                //Print responseObject 
               } 
              }]; 
     // Start the file upload. 
     [self.uploadTask resume]; 
    }]; 
} 

上述功能將繼續上傳圖像,當應用程序轉到後臺。所以,這不會給你內部服務器錯誤或任何超時錯誤。

希望它能幫助你。如果您對上述代碼有疑問,請隨時提問。