一文讲清:Android音频焦点

简介

1、概念

两个或两个以上的 Android 应用可同时向同一输出流播放音频,系统会将所有音频流混合在一起,为了避免所有音乐应用同时播放,Android 引入了“音频焦点”的概念。 一次只能有一个应用获得音频焦点。

2、使用

当你的应用需要输出音频时,需要请求获得音频焦点,获得焦点后,就可以播放声音了。当其他应用请求焦点时,会抢占你持有的音频焦点,此时,你的应用应暂停播放或降低音量,以便于用户听到新的音频源。

注:音频焦点策略是Android的规范化要求,但是并不是Audio出声的必要条件

使用方法

1、使用规则

  • 在即将开始播放之前调用 requestAudioFocus(),并验证调用是否返回 AUDIOFOCUS_REQUEST_GRANTED。即将开始播放,即在你代码逻辑中明确意图需要开始播放媒体时,需要调用。
  • 设置AudioFocusListener监听焦点状态,在其他应用获得音频焦点时,停止或暂停播放,或降低音量。
  • 播放停止且界面不在前台显示时,可抛弃音频焦点。(抛焦点策略需要慎重,需要结合系统整体音频策略)

2、关键代码

Android O 之前的焦点申请
 

java

代码解读

复制代码

AudioManager mAudioManager = (AudioManager) mContext.getSystemService(Context.AUDIO_SERVICE); int focusRequest = mAudioManager.requestAudioFocus( audioFocusListener, // 音频焦点监听器 AudioManager.STREAM_MUSIC, AudioManager.AUDIOFOCUS_GAIN); switch (focusRequest) { case AudioManager.AUDIOFOCUS_REQUEST_FAILED: // 请求焦点失败 -> 不允许播放 break; case AudioManager.AUDIOFOCUS_REQUEST_GRANTED: // 请求焦点成功 -> 开始播放 break; }

Android O 以及之后的版本上使用

整理了这份Java面试笔记包括了:Java面试、Spring、JVM、MyBatis、Redis、MySQL、并发编程、微服务、Linux、Springboot、SpringCloud、MQ、Kafka 面试专题

需要全套面试笔记【点击此处】即可免费获取

scss

代码解读

复制代码

val playbackAttributes = AudioAttributes.Builder() .setUsage(AudioAttributes.USAGE_MEDIA) .setContentType(AudioAttributes.CONTENT_TYPE_MUSIC) .build() mFocusRequest = AudioFocusRequest.Builder(AudioManager.AUDIOFOCUS_GAIN) .setAudioAttributes(playbackAttributes) .setAcceptsDelayedFocusGain(true) // 是否支持延时获取焦点 .setOnAudioFocusChangeListener(audioFocusListener, Handler(Looper.getMainLooper())) .build() // request接口需要用到mFocusRequest作为参数,mFocusRequest可一开始初始化一次 val focusRequest = mAudioManager.requestAudioFocus(mFocusRequest) if (focusRequest == AudioManager.AUDIOFOCUS_REQUEST_DELAYED) { // 申请delay了,需要处理焦点随后分发来的播放 } else if (focusRequest == AudioManager.AUDIOFOCUS_REQUEST_GRANTED) { // 申请成功,可以播放 } else if(focusRequest == AudioManager.AUDIOFOCUS_REQUEST_FAILED) { // 申请失败,不能播放 }

音频焦点的意图类型:

  • AUDIOFOCUS_GAIN 的使用场景:应用需要聚焦音频的时长会根据用户的使用时长改变,属于不确定期限。例如:多媒体播放或者播客等应用。
  • AUDIOFOCUS_GAIN_TRANSIENT_MAY_DUCK 的使用场景:应用只需短暂的音频聚焦,来播放一些提示类语音消息,或录制一段语音。例如:闹铃,导航等应用。
  • AUDIOFOCUS_GAIN_TRANSIENT 的使用场景:应用只需短暂的音频聚焦,但包含了不同响应情况,例如:电话、QQ、微信等通话应用。
  • AUDIOFOCUS_GAIN_TRANSIENT_EXCLUSIVE 的使用场景:同样您的应用只是需要短暂的音频聚焦。未知时长,但不允许被其它应用截取音频焦点。例如:录音软件。

requestResult — AudioManager.AUDIOFOCUS_REQUEST_DELAYED 说明:

假如当用户在通话中打开游戏,他们想玩游戏,但是当前还在打电话所以不想听到游戏声音。但是当他们通话结束的时候他们想听到游戏声音。如果这个应用支持延迟音频聚焦,会发生如下情况:

  • 当应用申请音频焦点的时候,会被拒绝并锁住,通话应用继续持有音频焦点,此时应用不能播放音频,因为应用申请焦点失败,但是申请的应用是游戏,可以正常继续操作,只是没有声音。
  • 当通话结束,之前申请的应用会被延迟分发音频焦点,这个授权是来自刚才申请音频聚焦被拒绝后锁住的那个请求,它只是被延迟一段时间后再授权,此时便可以开始恢复播放。

目前低于 Android O 的版本是不支持延迟音频焦点这个功能的。

音频焦点监听

一旦音频聚焦改变,应用要马上做出响应,它的状态可能在任何时间发生改变(丢失或重新获取),我们可以实现 OnAudioFocusChangeListener 的来响应状态改变:

 

kotlin

代码解读

复制代码

val audioFocusListener = object : AudioManager.OnAudioFocusChangeListener { override fun onAudioFocusChange(focusChange: Int) { Timber.i("$TAG, onAudioFocusChange, state: $focusChange") when (focusChange) { AudioManager.AUDIOFOCUS_GAIN -> { handleAudioGain() } AudioManager.AUDIOFOCUS_LOSS -> { handleAudioLoss() } AudioManager.AUDIOFOCUS_LOSS_TRANSIENT -> { handleAudioLoss(true) } AudioManager.AUDIOFOCUS_LOSS_TRANSIENT_CAN_DUCK -> {} // ignore else -> {} } } }

状态说明:

  • AUDIOFOCUS_GAIN:重新获取音频焦点。
  • AUDIOFOCUS_LOSS:永久性失去音频焦点,则其他应用会播放音频。您的应用应立即暂停播放,清理资源;因为它不会收到 AUDIOFOCUS_GAIN 回调。
  • AUDIOFOCUS_ LOSS_TRANSIENT:暂时失去音频焦点,但是很快就会重新获得,在此状态应该暂停所有音频播放,但是不能清除资源,
  • AUDIOFOCUS_ LOSS_TRANSIENT _CAN_DUCK:暂时失去音频焦点,应用应该降低音量(一般系统底层实现),允许持续播放音频(以很小的声音),不需要完全停止播放。
焦点抛弃
 

kotlin

代码解读

复制代码

val result = mAudioManager.abandonAudioFocusRequest(mFocusRequest) // 0: 失败, 1: 成功 Timber.i("$TAG, abandonAudioFocus end, result: $result")

3、正常接入Android焦点策略后的播放场景:

4、常见日志分析

关键字:QQAudioFocus(自己应用的音频的TAG)、MediaFocusControl、CarAudioFocus(车载)

By the way :车载媒体中心“焦点”

结论: 车载媒体中心的音源抢占和通知只是决定当前系统关联模块的媒体信息展示,并不能作为各个媒体源的播放暂停控制依据。

成功

其他

失败

用户播放意图

尝试申请
Android焦点

requestAudioFocus

获取AudioFocus

结束

Psd

Csd

歌词服务

...

代码示例

AudioFocusManager代码示例

可以通过以下代码来对AudioFocus进行管理:

 

Kotlin

代码解读

复制代码

package com.max.audiofocus import android.content.Context import android.media.AudioAttributes import android.media.AudioFocusRequest import android.media.AudioManager import android.os.Handler import android.os.Looper import timber.log.Timber /** * @author MaChao * @Description 音频焦点管理单例类 */ object AudioFocusManager { private const val TAG = "AudioFocusManager" private val mAudioManager: AudioManager = (MusicApplication.mApp.getSystemService(Context.AUDIO_SERVICE)) as AudioManager private val mFocusRequest: AudioFocusRequest private var isHasFocus = false private var isPauseByLoss = false init { val audioFocusListener = AudioManager.OnAudioFocusChangeListener { focusChange -> Timber.i("$TAG, onAudioFocusChange, state: $focusChange") when (focusChange) { AudioManager.AUDIOFOCUS_GAIN -> { handleAudioGain() } AudioManager.AUDIOFOCUS_LOSS -> { handleAudioLoss() } AudioManager.AUDIOFOCUS_LOSS_TRANSIENT -> { handleAudioLoss(true) } AudioManager.AUDIOFOCUS_LOSS_TRANSIENT_CAN_DUCK -> {} // ignore else -> {} } } val playbackAttributes = AudioAttributes.Builder() .setUsage(AudioAttributes.USAGE_MEDIA) .setContentType(AudioAttributes.CONTENT_TYPE_MUSIC) .build() mFocusRequest = AudioFocusRequest.Builder(AudioManager.AUDIOFOCUS_GAIN) .setAudioAttributes(playbackAttributes) .setAcceptsDelayedFocusGain(true) // 是否支持延时获取焦点 .setOnAudioFocusChangeListener(audioFocusListener, Handler(Looper.getMainLooper())) .build() } /** * 申请音频焦点 * @param forceRequest Boolean 是否强制调用AudioService申请(系统音频经常挂,导致异常焦点Loss了不会分配过来) * @return Boolean */ fun requestAudioFocus(forceRequest: Boolean = true): Boolean { if (!forceRequest && isHasFocus) { Timber.i("$TAG, requestAudioFocus, qqmusic already has Focus") isPauseByLoss = false return true } val result = mAudioManager.requestAudioFocus(mFocusRequest) // 0: 失败, 1: 成功, 2: delay Timber.i("$TAG, requestAudioFocus end, result: $result") if (result == AudioManager.AUDIOFOCUS_REQUEST_DELAYED) { // 申请delay了一会会通过audioListener分过来 isPauseByLoss = true } else if (result == AudioManager.AUDIOFOCUS_REQUEST_GRANTED) { isPauseByLoss = false } isHasFocus = result == AudioManager.AUDIOFOCUS_REQUEST_GRANTED return isHasFocus } /** * 当前是否已经有焦点 * @return Boolean */ fun isCurrHasFocus(): Boolean { return isHasFocus } /** * 抛弃音频焦点 * @return Boolean 是否抛成功 */ fun abandonAudioFocus(): Boolean { val result = mAudioManager.abandonAudioFocusRequest(mFocusRequest) // 0: 失败, 1: 成功 Timber.i("$TAG, abandonAudioFocus end, result: $result") isHasFocus = false isPauseByLoss = false return result == AudioManager.AUDIOFOCUS_REQUEST_GRANTED } /** * 处理焦点获取 */ private fun handleAudioGain() { isHasFocus = true if (isPauseByLoss) { PlayManager.resume() } isPauseByLoss = false } /** * 处理焦点丢失 * @param isShortLoss Boolean 是否是短暂丢失 */ private fun handleAudioLoss(isShortLoss: Boolean = false) { isHasFocus = false if (PlayManager.isPlaying()) { PlayManager.pause() isPauseByLoss = true } if (!isShortLoss) { // 如果是长丢失,抛掉当前焦点 abandonAudioFocus() } } }

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值