NSMutableURLRequest и проблема с памятью - PullRequest
1 голос
/ 17 октября 2011

Доброе утро, Я разрабатываю приложение, которое загружает некоторые файлы; нет никаких реальных проблем, но есть только одна большая проблема: если я пытаюсь загрузить 1 ГБ файлов, он выделяет 1 ГБ памяти, и вы можете понять, что это не очень хорошая вещь.

Вот мой код: я выделяю данные как NSData, а затем использую NSMutableURLRequest + NSURLConnection для их загрузки; мой вопрос: есть ли способы сделать это без необходимости выделять всю память? Я искал, но ничего не нашел ...

_uploadDataFile - это экземпляр подкласса NSData, выделенный ранее.

NSMutableURLRequest *request = [[[NSMutableURLRequest alloc] initWithURL:[NSURL URLWithString:@"http://my.server/page"]
                                                            cachePolicy:NSURLRequestReloadIgnoringCacheData
                                                        timeoutInterval:60.0] autorelease];

NSData *dataFile;

if (_uploadDataFile == nil) {
    NSError *error = nil;
    dataFile = [[[NSData alloc] initWithContentsOfFile:@"pathToFile" options:NSDataReadingUncached error:&error] autorelease];

    if (error) {
        NSLog(@"Error %i: %@", [error code], [error localizedDescription]);
        [self fermaUpload]; //Stop Upload
        return;
    }
}

else 
    dataFile = [_uploadDataFile file];

NSMutableDictionary *dictPost = [[[NSMutableDictionary alloc] init] autorelease];
[dictPost setValue:dataFile forKey:@"uploads[]"];

NSData *mutData = [self dataForPOSTWithDictionary:dictPost boundary:@"0194784892923" nomeFile:[@"pathToFile" lastPathComponent]];

NSString *postLength = [NSString stringWithFormat:@"%d", [mutData length]];

[request setHTTPMethod:@"POST"];
[request setValue:postLength forHTTPHeaderField:@"Content-Length"];
[request setValue:@"multipart/form-data; boundary=0194784892923" forHTTPHeaderField:@"Content-Type"];
[request setHTTPBody:mutData];
[request setHTTPShouldHandleCookies:YES];

theConnection = [[NSURLConnection alloc] initWithRequest:request delegate:self];

и вот как - (NSData *) dataForPOSTWithDictionary: border: nomeFile: метод выглядит так:

- (NSData *)dataForPOSTWithDictionary:(NSDictionary *)aDictionary boundary:(NSString *)aBoundary nomeFile:(NSString *)nomeFile {
NSArray *myDictKeys = [aDictionary allKeys];
NSMutableData *myData = [NSMutableData dataWithCapacity:1];
NSString *myBoundary = [NSString stringWithFormat:@"--%@\r\n", aBoundary];

for(int i = 0;i < [myDictKeys count];i++) {
    id myValue = [aDictionary valueForKey:[myDictKeys objectAtIndex:i]];
    [myData appendData:[myBoundary dataUsingEncoding:NSUTF8StringEncoding]];
    //if ([myValue class] == [NSString class]) {
    if ([myValue isKindOfClass:[NSString class]]) {
        [myData appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"\r\n\r\n", [myDictKeys objectAtIndex:i]] dataUsingEncoding:NSUTF8StringEncoding]];
        [myData appendData:[[NSString stringWithFormat:@"%@", myValue] dataUsingEncoding:NSUTF8StringEncoding]];
    } else if(([myValue isKindOfClass:[NSURL class]]) && ([myValue isFileURL])) {
        [myData appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", [myDictKeys objectAtIndex:i], [[myValue path] lastPathComponent]] dataUsingEncoding:NSUTF8StringEncoding]];
        [myData appendData:[[NSString stringWithFormat:@"Content-Type: application/octet-stream\r\n\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
        [myData appendData:[NSData dataWithContentsOfFile:[myValue path]]];
    } else if(([myValue isKindOfClass:[NSData class]])) {
        [myData appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", [myDictKeys objectAtIndex:i], nomeFile] dataUsingEncoding:NSUTF8StringEncoding]];
        [myData appendData:[[NSString stringWithFormat:@"Content-Type: application/octet-stream\r\n\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
        [myData appendData:myValue];
    } // eof if()

    [myData appendData:[[NSString stringWithString:@"\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
} // eof for()
[myData appendData:[[NSString stringWithFormat:@"--%@--\r\n", aBoundary] dataUsingEncoding:NSUTF8StringEncoding]];

return myData; } // eof dataForPOSTWithDictionary:boundary:

Спасибо и извините за мой английский.

1 Ответ

3 голосов
/ 19 октября 2011

Самая интересная проблема!Существует два способа ввода данных тела в NSURLConnection/Request:

  • NSData
  • NSInputStream

К сожалению, ни то, что вам нужноиз коробки, так как вы хотите загрузить файл + некоторые данные формы.В идеале вы можете создать подкласс NSInputStream для предоставления данных, отформатированных по вашему усмотрению.Однако в последний раз, когда я проверял, NSURLConnection не может обрабатывать такие подклассы.Вместо этого сделайте что-то вроде:

  1. Создайте пару потоков, используя CFStreamCreateBoundPair().Это возвращает CFWriteStream и CFReadStream, которые по бесплатному мосту соединены с NSOutputStream и NSInputStream соответственно
  2. Установите себя в качестве делегата выходного потока
  3. -setHTTPBodyStream:при запросе URL с входным потоком
  4. Запустите соединение

Теперь необходимо периодически вызывать делегат выходного потока, запрашивая данные.Вам необходимо ввести данные POST, а затем фрагменты данных файла.Таким образом, весь файл никогда полностью не находится в памяти сразу.

...