调用AudioManager.startBluetoothSCO()时在 list 中以 API 级别 18 或更高级别为目标时,文档指出已建立原始音频连接,如果以 API 17 或更低级别为目标,则使用虚拟语音调用。
在 API 级别 20(Android L 预览版)之前,它工作正常,适用于任何 API。但是,当使用最新的 Android Lollipop build LPX13D 并以 API 级别 18 或更高版本为目标时,我遇到了以下堆栈跟踪崩溃:
E/AndroidRuntime(31705): Caused by: java.lang.NullPointerException: Attempt to invoke virtual method 'java.lang.String android.bluetooth.BluetoothDevice.getAddress()' 在空对象引用上 E/AndroidRuntime(31705): 在 android.os.Parcel.readException(Parcel.java:1546) E/AndroidRuntime(31705): 在 android.os.Parcel.readException(Parcel.java:1493) E/AndroidRuntime(31705): 在 android.media.IAudioService$Stub$Proxy.startBluetoothSco(IAudioService.java:1587) E/AndroidRuntime(31705): 在android.media.AudioManager.startBluetoothSco(AudioManager.java:1468)
如果我在 Android Lollipop 上以 API 级别 17 或更低级别为目标,一切都会按预期运行。
我相信问题的根源在于在文件 AudioService.java 第 2392 行中 API 级别 21 中发生的 Android 音频代码更改:
public void startBluetoothSco(IBinder cb, int targetSdkVersion) {
int scoAudioMode =
(targetSdkVersion < Build.VERSION_CODES.JELLY_BEAN_MR2) ?
SCO_MODE_VIRTUAL_CALL : SCO_MODE_UNDEFINED;
startBluetoothScoInt(cb, scoAudioMode);
}
看起来 SCO_MODE_UNDEFINED 应该改为 SCO_MODE_RAW。如果您查看该文件,您会发现在几个地方检查了 SCO_MODE_RAW,但实际上从未在任何地方传递过。
还有其他人遇到过这次崩溃吗?有谁知道比将目标 SDK 降级到 17 更好的解决方法吗?如果没有,请给bug report加注星标好吗?我向 Google 提交了申请,以增加它被查看的机会:-)
最佳答案
正如@xsveda 所写,如果没有连接耳机,您将在 Lollipop 上收到 NPE。
您可以先尝试检查蓝牙耳机连接情况:
mAudioManager.isWiredHeadsetOn()
如文档所述,isWiredHeadsetOn()
( doc link ) 已被删除,仅用于检查耳机是否已连接。
在此之后您可以使用startBluetoothSco()
连接。至于我,我使用了这段代码:
这个开始:
if(mAudioManager.isWiredHeadsetOn())
mAudioManager.startBluetoothSco();
这个停止:
if(mAudioManager.isBluetoothScoOn())
mAudioManager.stopBluetoothSco();
希望对您有所帮助。
关于android - AudioManager.startBluetoothSco() 在 Android Lollipop 上崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26642218/