我建议只是开始上传并继续发送数据。您还可以避免创建 250mb 缓冲区,方法是使用uploadTaskWithStreamedRequest
然后创建一个NSInputStream
子类,该子类一直提供更多数据,直到您告诉它停止。您可以实施URLSession:task:didSendBodyData:totalBytesSent:totalBytesExpectedToSend:
以监控上传进度(因此您可能可以监控数据发送的速度)。
无论如何,要创建上传请求:
@interface ViewController () <NSURLSessionDelegate, NSURLSessionTaskDelegate>
@property (nonatomic, strong) CustomStream *inputStream;
@end
@implementation ViewController
- (void)viewDidLoad
{
[super viewDidLoad];
self.inputStream = [[CustomStream alloc] init];
NSURL *url = [NSURL URLWithString:kURLString];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
[request setHTTPMethod:@"POST"];
NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration defaultSessionConfiguration];
NSURLSession *session = [NSURLSession sessionWithConfiguration:configuration delegate:self delegateQueue:nil];
NSURLSessionUploadTask *task = [session uploadTaskWithStreamedRequest:request];
[task resume];
// I don't know how you want to finish the upload, but I'm just going
// to stop it after 10 seconds
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(10.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
self.inputStream.finished = YES;
});
}
您显然必须实现适当的委托方法:
#pragma mark - NSURLSessionTaskDelegate
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didSendBodyData:(int64_t)bytesSent totalBytesSent:(int64_t)totalBytesSent totalBytesExpectedToSend:(int64_t)totalBytesExpectedToSend
{
NSLog(@"%lld %lld %lld", bytesSent, totalBytesSent, totalBytesExpectedToSend);
}
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task needNewBodyStream:(void (^)(NSInputStream *bodyStream))completionHandler
{
completionHandler(self.inputStream);
}
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
NSLog(@"%s: error = %@; data = %@", __PRETTY_FUNCTION__, error, [[NSString alloc] initWithData:self.responseData encoding:NSUTF8StringEncoding]);
}
#pragma mark - NSURLSessionDataDelegate
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition disposition))completionHandler
{
self.responseData = [NSMutableData data];
completionHandler(NSURLSessionResponseAllow);
}
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
{
[self.responseData appendData:data];
}
和CustomStream
:
static NSInteger const kBufferSize = 32768;
@interface CustomStream : NSInputStream
@property (nonatomic, readonly) NSStreamStatus streamStatus;
@property (nonatomic, getter = isFinished) BOOL finished;
@end
@interface CustomStream ()
@property (nonatomic) NSStreamStatus streamStatus;
@property (nonatomic) void *buffer;
@end
@implementation CustomStream
- (instancetype)init
{
self = [super init];
if (self) {
_buffer = malloc(kBufferSize);
NSAssert(_buffer, @"Unable to create buffer");
memset(_buffer, 0, kBufferSize);
}
return self;
}
- (void)dealloc
{
if (_buffer) {
free(_buffer);
self.buffer = NULL;
}
}
- (void)open
{
self.streamStatus = NSStreamStatusOpen;
}
- (void)close
{
self.streamStatus = NSStreamStatusClosed;
}
- (NSInteger)read:(uint8_t *)buffer maxLength:(NSUInteger)len
{
if ([self isFinished]) {
if (self.streamStatus == NSStreamStatusOpen) {
self.streamStatus = NSStreamStatusAtEnd;
}
return 0;
}
NSUInteger bytesToCopy = MIN(len, kBufferSize);
memcpy(buffer, _buffer, bytesToCopy);
return bytesToCopy;
}
- (BOOL)getBuffer:(uint8_t **)buffer length:(NSUInteger *)len
{
return NO;
}
- (BOOL)hasBytesAvailable
{
return self.streamStatus == NSStreamStatusOpen;
}
- (void)scheduleInRunLoop:(__unused NSRunLoop *)aRunLoop
forMode:(__unused NSString *)mode
{}
- (void)removeFromRunLoop:(__unused NSRunLoop *)aRunLoop
forMode:(__unused NSString *)mode
{}
#pragma mark Undocumented CFReadStream Bridged Methods
- (void)_scheduleInCFRunLoop:(__unused CFRunLoopRef)aRunLoop
forMode:(__unused CFStringRef)aMode
{}
- (void)_unscheduleFromCFRunLoop:(__unused CFRunLoopRef)aRunLoop
forMode:(__unused CFStringRef)aMode
{}
- (BOOL)_setCFClientFlags:(__unused CFOptionFlags)inFlags
callback:(__unused CFReadStreamClientCallBack)inCallback
context:(__unused CFStreamClientContext *)inContext {
return NO;
}
@end
我建议你参考 BJ Homer 的文章Subclassing NSInputStream来了解这个子类中一些神秘方法的背景NSInputStream
。