10

我正在尝试使用 AVAudioRecorder 的 averagePowerForChannel 方法来监视 iPad/iPhone 应用程序麦克风上的输入电平。我有一个回调,它在循环中轮询平均水平——在 iPhone 上它工作正常并返回合理的水平,但由于某种原因在 iPad 上它总是返回 -120.0。

这是我的一些设置代码:

- (void) setupMic {
if (micInput) {
    [micInput release];
    micInput = nil;
}
NSURL *newURL = [[NSURL alloc] initFileURLWithPath:@"/dev/null"];

NSMutableDictionary *recordSettings = [[NSMutableDictionary alloc] init];

[recordSettings setObject:[NSNumber numberWithInt:kAudioFormatAppleLossless] forKey: AVFormatIDKey];
[recordSettings setObject:[NSNumber numberWithFloat:22050.0] forKey: AVSampleRateKey];
//  [recordSettings setObject:[NSNumber numberWithInt:2] forKey:AVNumberOfChannelsKey];
[recordSettings setObject:[NSNumber numberWithInt:12800] forKey:AVEncoderBitRateKey];
[recordSettings setObject:[NSNumber numberWithInt:16] forKey:AVLinearPCMBitDepthKey];
[recordSettings setObject:[NSNumber numberWithInt: AVAudioQualityLow] forKey: AVEncoderAudioQualityKey];

micInput = [[AVAudioRecorder alloc] initWithURL:newURL settings:recordSettings error:nil];
//  [micInput setMeteringEnabled:YES];

[newURL release];
[recordSettings removeAllObjects];
[recordSettings release];
}

以及我的开始录制方法:

- (void) startRecording {
NSLog(@"startRecording!");
[micInput pause];
[micInput prepareToRecord];
micInput.meteringEnabled = YES;
[micInput record];
[micInput updateMeters];
levelTimer = [[NSTimer alloc] initWithFireDate:[NSDate dateWithTimeIntervalSinceNow:0.0] interval:0.03 target:self selector:@selector(levelTimerCallback:) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:levelTimer forMode:NSDefaultRunLoopMode];
}

和一些 levelTimer 回调:

- (void)levelTimerCallback:(NSTimer *)timer {
[micInput updateMeters];
double avgPowerForChannel = pow(10, (0.05 * [micInput averagePowerForChannel:0]));
[micSprite receiveInput:avgPowerForChannel];

NSLog(@"Avg. Power: %f", [micInput averagePowerForChannel:0]);

 ...

}

在 iPhone 上,NSLog 语句将返回合理的值,而 iPad 将始终返回 -120.0。

注意:我在 cocos2d 应用程序中使用它。出于某种原因,如果我在 iPad 上重新启动当前场景,麦克风电平将返回正确的值。

有人有什么建议吗?我在这里严重不知所措。谢谢!

4

5 回答 5

22

我遇到过同样的问题。我发现将类别设置为 AVAudioSessionCategoryPlayAndRecord 可以修复它:

NSError *error;
[[AVAudioSession sharedInstance] 
    setCategory:AVAudioSessionCategoryPlayAndRecord error:&error];

if (error) {
    NSLog(@"Error setting category: %@", [error description]);
}
于 2011-04-04T11:54:47.730 回答
1

是的,我也是。只有一个共享的 AVAudioSession。设置其类别会影响所有录音机和播放器,因此在应用程序的一个区域将类别设置为 AVAudioSessionCategoryPlay 例如会禁用其他区域的录音机。

于 2012-02-11T14:23:29.693 回答
0

另一个可能的原因很简单,请确保您正在调用 [recorder record];

那条线被意外删除了,我们花了一段时间才意识到这是仪表上返回 -120 值的原因。

高温高压

于 2012-07-07T07:05:39.130 回答
0

1)确保您有权限:

if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 7) {
    if([[AVAudioSession sharedInstance] respondsToSelector:@selector(requestRecordPermission:)])
    {
        [[AVAudioSession sharedInstance] requestRecordPermission:^(BOOL granted) {
            NSLog(@"permission : %d", granted);
            if(granted)[self setupMic];// record ...
        }];
    }
}

2)确保您的路径是合法的(/dev/null ok??):

NSString* dir=[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject];
NSArray *pathComponents = [NSArray arrayWithObjects:dir,fileName,nil];
recordedAudioURL = [NSURL fileURLWithPathComponents:pathComponents];

3) 激活音频会话

 [audioSession setActive:YES error:&error];

4)检查所有错误和返回值,包括

BOOL ok=[micInput prepareToRecord];
ok=ok &&[micInput record];
于 2015-12-10T20:51:49.343 回答
0

您需要初始化 AVAudioSession (应该从 didMoveToview 方法完成,这可能是问题的原因)

let session = AVAudioSession.sharedInstance()
    try session.setCategory(AVAudioSessionCategoryPlayAndRecord)
    try session.overrideOutputAudioPort(AVAudioSessionPortOverride.speaker)
    try session.setActive(true)
try recorder = AVAudioRecorder(url: getDocumentsDirectory(), settings: settings)

并启动录音机

func start() {
    recorder?.record()
    recorder?.updateMeters()
}

并通过调用获得分贝

let decibels = recorder.averagePower(forChannel: 0)

从 -120(最小值)到 0。例如,某些咖啡馆的噪音水平是 -20

这是更详细的示例http://www.mikitamanko.com/blog/2017/04/15/swift-how-to-get-decibels/注意:也使用更新方法。

于 2017-04-15T17:37:56.093 回答