1

早上好,我正在开发一个上传一些文件的应用程序;没有真正的问题,但只有一个大问题:如果我尝试上传 1gb 的文件,它会分配 1gb 的内存,你可以理解这不是一件好事。

这是我的代码:我将数据分配为 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:boundary: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:

谢谢你,对不起我的英语。

4

1 回答 1

3

一个最有趣的问题!有两种方法可以将身体数据输入NSURLConnection/Request

  • NSData
  • NSInputStream

不幸的是,你想要的也不是开箱即用的,因为你想上传一个文件+一些表单数据。理想情况下,您可以进行子类NSInputStream化以提供您需要的格式化数据。但是,我上次检查时,NSURLConnection无法处理此类子类。相反,请执行以下操作:

  1. 使用创建一对流CFStreamCreateBoundPair()。这将返回 aCFWriteStreamCFReadStream,它们是免费桥接到NSOutputStreamNSInputStream,分别
  2. 将自己设置为输出流的委托
  3. -setHTTPBodyStream:在带有输入流的 URL 请求上
  4. 开始连接

现在应该定期调用输出流的委托,请求数据。您需要输入 POST 数据,然后是文件数据块。因此,整个文件永远不会一次完全在内存中。

于 2011-10-19T10:59:40.470 回答