1

我对 iPhone 开发有点陌生,所以要温柔!我支持从 URL 文件流加载 wav 文件并通过 AudioQueue 播放的应用程序。

我们在另一个线程中运行一个连续循环,如果我们检测到它没有使用缓冲区,并且输入 FileStream 已到达其末尾,则停止队列。反过来,我们通过检查 是否长度为 0来检测 FileStream 是否在流的waitForDataInBackgroundAndNotify回调中结束。NSFileHandleDataAvailableNotificationavailableData

这适用于 iOS 3.0 - 我们在文件末尾收到 0 个可用数据的通知 - 但在 iOS 4.0 上,我们似乎没有在文件末尾收到回调。这发生在 OS 4.0 设备上,与目标 OS 版本无关。

两个版本之间的 API 是否发生了变化?我现在如何检测文件的结尾?

希望相关的代码:

数据可用回调:

- (void)readFileData:(NSNotification *)notification
{
  @try
  {
    NSData *data = [[notification object] availableData];

    if ([data length] == 0 && self.audioQueueState != AQS_END)
    {
      /***********************************************************************/
      /* We've hit the end of the data but it's possible that more may be    */
      /* appended to the file (if we're still downloading it) so we need to  */
      /* wait for the availability of more data.                             */
      /***********************************************************************/
      [self setFileStreamerState:FSS_END];
      [[notification object] waitForDataInBackgroundAndNotify];
    }
    else if (self.audioQueueState == AQS_END)
    {
      TRC_DBG(@"ignore read data as ending");
    }
    else
    {
      NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];

      TRC_DBG(@"Read %d bytes", [data length]);

      [self setFileStreamerState:FSS_DATA];

      if (discontinuous)
      {
        TRC_DBG(@"AudioFileStreamParseBytes %d bytes, discontinuous", [data length]);
        err = AudioFileStreamParseBytes(audioFileStream, [data length], [data bytes], kAudioFileStreamParseFlag_Discontinuity);
        discontinuous = NO;
      }
      else
      {
        TRC_DBG(@"AudioFileStreamParseBytes %d bytes, continuous", [data length]);
        err = AudioFileStreamParseBytes(audioFileStream, [data length], [data bytes], 0);
      }

      /***********************************************************************/
      /* If error then get out, otherwise wait again for more data.          */
      /***********************************************************************/
      if (err != 0)
      {
        [self failWithErrorCode:AS_FILE_STREAM_PARSE_BYTES_FAILED];
      }
      else
      {
        [[notification object] waitForDataInBackgroundAndNotify];
      }

      [pool release];
    }
  }
  @catch (NSException *exception)
  {
    TRC_ERR(@"Exception: %@", exception);
    TRC_ERR(@"Exception reason: %@", [exception reason]);
    //[self failWithErrorCode:AS_FILE_AVAILABLE_DATA_FAILED];
  }
}
4

0 回答 0