10

我看到一个问题CFReadStreamRead,其中作为流式文件上传的一部分,永远不会返回。

似乎只发生在 iOS7 上——并且在针对物理设备进行调试时比在模拟器中更频繁——或者至少,它在那里更为明显。

我们有一个文件的 HTTP(或 HTTPS,问题发生在任何一种方式,使用本地托管或远程服务器)POST,通过直线阻塞(非事件驱动)CFNetwork 调用。这是 C 代码调用这个处理程序的必要条件;没有回调的规定。

这很好,网络调用发生在后台线程和/或通过异步调度。

有问题的网络代码归结为(为简洁起见删除错误处理):

CFReadStreamRef upload = CFReadStreamCreateWithFile(
  kCFAllocatorDefault, upload_file_url);
CFRelease(upload_file_url);
CFReadStreamOpen(upload);

CFReadStreamRef myReadStream = CFReadStreamCreateForStreamedHTTPRequest(
  kCFAllocatorDefault, myRequest, upload);

CFReadStreamOpen(myReadStream);

CFIndex numBytesRead = CFReadStreamRead(myReadStream, buf, sizeof(buf));

// etc.

就其本身而言,这段代码想在 iOS7 下立即挂起。usleep如果我在它之前添加一个带有一些调用的循环(CFReadStreamHasBytesAvailable一路检查),它几乎总是会成功。每隔几百次尝试,它仍然会失败,永远不会返回。同样,主线程不受影响。

我希望 GM 能清除这种行为,但它仍然存在。

添加一个 runloop/callback 方法来监视可用字节事件没有任何效果 - 当调用挂起时,也看不到任何事件。

关于为什么会发生这种情况或如何防止这种情况的任何建议?还有其他人CFReadStream在 iOS 7 下看到不同的行为吗?

4

1 回答 1

0

我已经尝试过这种讨厌的解决方法,它对我有用,问题是我正在从服务器请求增量值,所以如果出现问题,我只是获取新的增量值,一般情况下它不会工作(在日志中我看到有时会出现超时)。至少这可以防止形成永久线程阻塞,并有机会以某种方式处理这个问题:

NSInteger readStreamReadWorkaround(CFReadStreamRef readStrem, UInt8 *buffer, CFIndex bufferLength) {
  static dispatch_once_t onceToken;
  static BOOL isProblematicOs = YES;
  dispatch_once(&onceToken, ^{
    isProblematicOs = [[UIDevice currentDevice].systemName compare: @"7.0" options: NSNumericSearch]!=NSOrderedAscending;
  });

  NSInteger readBytesCount = -2;

  if (isProblematicOs) {
    CFStreamStatus sStatus = CFReadStreamGetStatus(readStrem);
    NSDate *date = [NSDate date];
    while (YES) {
      if(CFReadStreamHasBytesAvailable(readStrem)) {
        readBytesCount = CFReadStreamRead(readStrem, buffer, bufferLength);
        break;
      }
      sStatus = CFReadStreamGetStatus(readStrem);
      if (sStatus!=kCFStreamStatusOpen && sStatus !=kCFStreamStatusAtEnd
          || [date timeIntervalSinceNow]<-15.0) {
        break;
      }
      usleep(50000);
    }
  } else {
    readBytesCount = CFReadStreamRead(readStrem, buffer, sizeof(buffer));
  }
  return readBytesCount;
}

我不喜欢这个解决方案,但到目前为止我还没有看到替代方案。

于 2013-10-04T12:57:12.550 回答