由旧版本环信对接之后没有声音,探索AVAudioSession的Category和Option

116 阅读3分钟

前言

接手了一个有一些历史的项目,这个项目当中的环信比较老,也有一段时间没有运营了,再次运行起来之后,发现音频拨打通之后没有任何的声音。查来查去,也发了工单,找不到问题的原因。然后重新梳理代码,发现里面有AVAudioSession这么一个玩意儿,经过一番查找学习,最后解决了这个问题。现在记录一下。

正文

首先讲一下问题原因以及解决,原因在于AVAudioSession设置的AVAudioSessionCategory的模式不对。解决方案为,

     [[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayAndRecord withOptions:AVAudioSessionCategoryOptionMixWithOthers error:nil];

那么问题解决了,为什么会出现这个问题,需要讲解一下,或许你在使用环信的时候并没有遇到这个问题,但是当遇到了,希望可以帮助你提供思路。

AVAudioSessionCategory

首先讲一下AVAudioSessionCategory,这个枚举的设置会影响到当前应用的播放与录音相关配置。值得一提的是,改category设置之后,是可以根据使用场景灵活切换的,当只需要某一种模式的时候,是可以通过设置来进行变更的。以下为该category的列举,可以根据实际场景灵活判断:

AVAudioSessionCategoryAmbient

用于非以语音为主的应用,使用这个category的应用随着静音键屏幕关闭静音。并且不会中止其它应用播放声音,可以和其它自带应用如iPod,safari等同时播放声音。**注意:**该Category无法在后台播放声音

AVAudioSessionCategorySoloAmbient

和AVAudioSessionCategoryAmbient相同,但是区别在于会中止其他应用的播放。

AVAudioSessionCategoryPlayback

用于以语音为主的应用,使用这个category的应用不会随着静音键屏幕关闭而静音。可在后台播放声音

AVAudioSessionCategoryRecord

用于需要录音的应用,设置该category后,除了来电铃声,闹钟或日历提醒之外的其它系统声音都不会被播放。该Category只提供单纯录音功能。

AVAudioSessionCategoryPlayAndRecord

用于既需要播放声音需要录音的应用,语音聊天应用(如微信)应该使用这个category。该Category提供录音和播放功能。如果你的应用需要用到iPhone上的听筒,该category是你唯一的选择,在该Category下声音的默认出口为听筒(在没有外接设备的情况下)。

AVAudioSessionCategoryOptions

主要用于控制当前应用和其他应用中间的关系,枚举如下:

AVAudioSessionCategoryOptionMixWithOthers

设置这个选项在激活会话时,不会打断其他应用程序的音频播放,在以下的选项中可用AVAudioSessionCategoryPlaybackAVAudioSessionCategoryPlayAndRecordAVAudioSessionCategoryMultiRoute

AVAudioSessionCategoryOptionDuckOthers

这个选项在激活会话时降低其他程序的音频播放声音,在以下的选项中可用AVAudioSessionCategoryPlayAndRecordAVAudioSessionCategoryRecord 主要是体现当前音频的重要性,例如使导航音量凸出于音乐播放的音量,特别关注的音量高于当前视频的音量等操作。

AVAudioSessionCategoryOptionAllowBluetooth

允许可免提蓝牙设备可使用输入通道

AVAudioSessionCategoryOptionDefaultToSpeaker

设置这个选项在没有其他通道的时候默认选择内置扬声器,类似于默认免提功能

AVAudioSessionCategoryOptionInterruptSpokenAudioAndMixWIthOthers

iOS9新加入的功能,当你的app偶尔的使用音频播放时打断其他应用,使其静音,并自动使用麦克风以及使用完毕后自动恢复其他应用的播放(可以在以下的选项中可用AVAudioSessionCategoryPlaybackAVAudioSessionCategoryPlayAndRecordAVAudioSessionCategoryMultiRoute 如果设置了这个选项,在关闭这个session时需要:

[myAudioSession setActive: NO with Options: AVAudioSessionSetActiveOptionNotifyOthersOnDeactivation error: <#Your error object, or nil for testing#>];

这句代码是为了确保执行当前session之前被打断的正在播放的 audio app能都收到resume消息,从而恢复到正常播放的状态。