说明 我需要在HTTP POST参数中发送一个base 64编码的字符串,表示要上传的文件。
当前方法:我正在使用ASIFormDataRequest并将我的文件编码为base64字符串,如下所示:
NSData *data = [[NSFileManager defaultManager] contentsAtPath:@"my path here"];
NSString *base64Data = [data encodeBase64ForData];
然而,当上传大文件时,应用程序内存不足并死于可怕的死亡!
建议的解决方案:现在有人如何进行,比如从磁盘读取文件,将其转换为块中的base64字符串,并将转换后的base64字符串附加到HTTP请求中的参数?
有效避免将整个文件读入内存的任何方式。
非常感谢任何输入!
我的http请求代码已完整:
NSURL *url = [NSURL URLWithString:requestProgressURLString];
NSData *data = [[NSFileManager defaultManager] contentsAtPath:evidence.uri];
NSString *base64Data = [data encodeBase64ForData];
NSURL *fileURL = [[NSURL alloc] initWithString:evidence.uri];
request = [ASIFormDataRequest requestWithURL:url];
request.shouldAttemptPersistentConnection = false;
[request setShowAccurateProgress:YES];
[request setPostValue:accessToken forKey:@"AccessToken"];
[request setPostValue:[[fileURL path] lastPathComponent] forKey:@"Filename"];
[request setPostValue:evidence.name forKey:@"Title"];
[request setPostValue:base64Data forKey:@"FileData"]; //wants string value (base64 encoded)
[request addRequestHeader:@"Content-Type" value:@"application/x-www-form-urlencoded"];
[request setTimeOutSeconds:60];
[request setUploadProgressDelegate:self];
[request setDownloadProgressDelegate:self];
[request setDelegate:self];
[request setDidFinishSelector:@selector(requestDone:)];
[request setDidFailSelector:@selector(requestWentWrong:)];
答案 0 :(得分:3)
我会把这个问题分成两部分:
<强> 1。将大文件编码为块中的Base64
的 2。上传该文件
<强> 1。将大文件编码为以块为单位的Base64:
查找将NSData
个实例编码为Base64的代码。然后使用一个NSInputStream
以块的形式读取数据,对其进行编码并使用NSOutputStream
写入临时文件。我发现了这个问题/答案:Is it possible to base64-encode a file in chunks?使块大小为3的倍数,它应该可以工作。您可能希望将其移至后台线程。
重要提示:这将创建一个比原始文件大的临时文件。那个时刻设备上必须有足够的空间。开始上传后,您可以将其删除。
<强> 2。上传该文件
看来你已经完成了这件事。您可以在示例中使用ASIFormDataRequest
。由于它制作了文件的私有副本(它构建了多部分POST文件),因此您可以删除副本。
如果要通过并行化这些步骤来优化它,则需要找到另一种上传多部分文件的方法。
答案 1 :(得分:0)
不是在一次点击中读取文件,而是需要以块的形式阅读
您可以使用输入流来执行此操作 Apple在here
上有一些信息基本上,您需要创建一个指向您文件的NSInputStream,如示例
中所示- (void)setUpStreamForFile:(NSString *)path {
// iStream is NSInputStream instance variable
iStream = [[NSInputStream alloc] initWithFileAtPath:path];
[iStream setDelegate:self];
[iStream scheduleInRunLoop:[NSRunLoop currentRunLoop]
forMode:NSDefaultRunLoopMode];
[iStream open];
}
然后,您可以使用NSStreamDelegate方法以块的形式读取数据 - (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode
Apple再次举例说明了这个
- (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode {
switch(eventCode) {
case NSStreamEventHasBytesAvailable:
{
if(!_data) {
_data = [[NSMutableData data] retain];
}
uint8_t buf[1024];
unsigned int len = 0;
len = [(NSInputStream *)stream read:buf maxLength:1024];
if(len) {
[_data appendBytes:(const void *)buf length:len];
// bytesRead is an instance variable of type NSNumber.
[bytesRead setIntValue:[bytesRead intValue]+len];
// DO SOMETHING with DATA HERE
} else {
NSLog(@"no buffer!");
}
break;
}
您选择的块大小需要在内存性能与网络性能之间取得平衡,因为HTTP标头显然存在一些开销
您还可以通过直接发送NSData进一步优化(因此节省base64开销)查看此帖子