转:https://baijiahao.baidu.com/s?id=1708682753422314151&wfr=spider&for=pc
音频输出作为硬件资源,对于iOS系统来说是唯一的,需要通过“AVAudioSession”这个系统级全局对象对各个App进行控制。
常见音频场景
是进行录音还是播放?
当系统静音键按下时该如何表现?
是从扬声器还是从听筒里面播放声音?
插拔耳机后如何表现?
来电话/闹钟响了后如何表现?
其他音频App启动后如何表现?
通过“AVAudioSession”可以做什么操作
为app选择输入输出的路由:通过扬声器或者听筒播放--可以手动设置通过手机上扬声器/听筒或者耳机播放--系统自动切换,插入耳机默认耳机
为app选择输入的路由:通过手机麦克风或者耳机麦克风采集-系统自动切换,插入耳机默认耳机
协调音频播放的app之间的关联,以及系统的声音处理
处理被其他app打断后的情况
录音或者播放音乐
AVAudioSession默认行为
可以进行播放,但是不能进行录制。
当用户将手机上的静音拨片拨到“静音”状态时,此时如果正在播放音频,那么播放内容会被静音。
当用户按了手机的锁屏键或者手机自动锁屏了,此时如果正在播放音频,那么播放会静音并被暂停。
如果你的App在开始播放的时候,此时QQ音乐或其他App正在播放,那么其他播放器会被静音并暂停。
系统会在app启动的时候,激活这个全局唯一的AVAudioSession
音频模式七大场景
该场景由7个类别分别控制:
默认是:CategorySoloAmbient(独占播放)
AVAudioSessionCategoryAmbient : 只用于播放音乐时,并且可以和QQ音乐同时播放,比如玩游戏的时候还想听QQ音乐的歌,那么把游戏播放背景音就设置成这种类别。同时,当用户锁屏或者静音时也会随着静音,这种类别基本适用所有App的背景场景。
AVAudioSessionCategorySoloAmbient: 也是只用于播放,但是和"AVAudioSessionCategoryAmbient"不同的是,用了它就别想听QQ音乐了,比如不希望QQ音乐干扰的App,类似节奏大师。同样当用户锁屏或者静音时也会随着静音,锁屏了就玩不了节奏大师了。
AVAudioSessionCategoryPlayback: 如果锁屏了还想听声音怎么办?用这个类别,比如App本身就是播放器,同时当App播放时,其他类似QQ音乐就不能播放了。所以这种类别一般用于播放器类App
AVAudioSessionCategoryRecord: 有了播放器,肯定要录音机,比如微信语音的录制,就要用到这个类别,既然要安静的录音,肯定不希望有QQ音乐了,所以其他播放声音会中断。想想微信语音的场景,就知道什么时候用他了。
AVAudioSessionCategoryPlayAndRecord: 如果既想播放又想录制该用什么模式呢?比如VoIP,打电话这种场景,PlayAndRecord就是专门为这样的场景设计的 。
AVAudioSessionCategoryMultiRoute: 想象一个DJ用的App,手机连着HDMI到扬声器播放当前的音乐,然后耳机里面播放下一曲,这种常人不理解的场景,这个类别可以支持多个设备输入输出。
AVAudioSessionCategoryAudioProcessing: 主要用于音频格式处理,一般可以配合AudioUnit进行使用
上面介绍的七大类别,可以认为是设定了七种主场景,而这七类肯定是不能满足开发者所有的需求的。AVAudioSession提供的方法是,首先定下7种类别中的一种基调,再进行微调。AVAudioSession为每种Category都提供了些许选项来进行微调。
类别的选项
AVAudioSessionCategoryOptionMixWithOthers : 如果确实用的AVAudioSessionCategoryPlayback实现的一个背景音,但是又想和QQ音乐并存,可以在AVAudioSessionCategoryPlayback类别下在设置这个选项,就可以实现共了。
AVAudioSessionCategoryOptionDuckOthers:在实时通话的场景,比如QQ音乐,当进行视频通话的时候,会发现QQ音乐自动声音降低了,此时就是通过设置这个选项来对其他音乐App进行了压制。
AVAudioSessionCategoryOptionAllowBluetooth:如果要支持蓝牙耳机电话,则需要设置这个选项
AVAudioSessionCategoryOptionDefaultToSpeaker: 如果在VoIP模式下,希望默认打开免提功能,需要设置这个选项
七大模式
AVAudioSessionModeDefault: 每种类别默认的就是这个模式。
AVAudioSessionModeVoiceChat:主要用于VoIP场景,此时系统会选择最佳的输入设备,比如插上耳机就使用耳机上的麦克风进行采集。同时会设置类别的选项为"AVAudioSessionCategoryOptionAllowBluetooth"从而支持蓝牙耳机。
AVAudioSessionModeVideoChat : 主要用于视频通话,比如QQ视频、FaceTime。系统会选择最佳的输入设备,比如插上耳机就使用耳机上的麦克风进行采集并且会设置类别的选项为"AVAudioSessionCategoryOptionAllowBluetooth" 和 "AVAudioSessionCategoryOptionDefaultToSpeaker"。
AVAudioSessionModeGameChat : 适用于游戏App的采集和播放,比如“GKVoiceChat”对象,一般不需要手动设置
每个模式有其适用的类别,可以根据具体需要做对应的组合。
音频通知
系统中断通知
如果app正在视频通话,突然来电话了、闹钟响了等,最常用的场景是先暂停,待恢复的时候再继续。app可以通过监听系统消息:AVAudioSessionInterruptionNotification来处理。系统消息userInfo主要包含两个键:
AVAudioSessionInterruptionTypeKey: 取值为AVAudioSessionInterruptionTypeBegan表示中断开始,我们应该暂停播放和采集,取值为AVAudioSessionInterruptionTypeEnded表示中断结束,我们可以继续播放和采集。
AVAudioSessionInterruptionOptionKey: 当前只有一种值AVAudioSessionInterruptionOptionShouldResume表示此时也应该恢复继续播放和采集。
其他app占用通知
其他App占据AudioSession的时候用AVAudioSessionSilenceSecondaryAudioHintNotification来进行通知。其回调回来的userInfo键为:AVAudioSessionSilenceSecondaryAudioHintTypeKey
AVAudioSessionSilenceSecondaryAudioHintTypeBegin:表示其他App开始占据Session
AVAudioSessionSilenceSecondaryAudioHintTypeEnd:表示其他App开始释放Session
外设改变通知
默认情况下,AVAudioSession会在App启动时选择一个最优的输出方案,比如插入耳机的时候,就用耳机。但是这个过程中,用户可能拔出耳机,app可以通过监听系统消息:AVAudioSessionRouteChangeNotification来处理。
针对视频通话场景设置
类别:AVAudioSessionCategoryPlayAndRecord类别的选项:AVAudioSessionCategoryOptionAllowBluetoothAVAudioSessionCategoryOptionDefaultToSpeaker模式:AVAudioSessionModeVideoChat