Android,插入耳机后如何将音频路由到扬声器?

Posted

技术标签:

【中文标题】Android,插入耳机后如何将音频路由到扬声器?【英文标题】:Android, how to route the audio to speakers when headphones inserted? 【发布时间】:2016-01-30 12:11:23 【问题描述】:

我正在寻找一种从 android 设备的扬声器播放音频的方法,即使插入了耳机也是如此。

实际上,典型的行为是,当插入耳机时,扬声器不会输出任何音频。但是,即使插入耳机,某些应用程序(例如默认时钟应用程序 (com.google.android.deskclock))也能够将音频路由到扬声器。

如何以编程方式获取此行为?

我正在寻找一种可以(至少)在运行 KitKat (Android 4.4) 的 Nexus 5 设备上运行的解决方案

谢谢。

【问题讨论】:

【参考方案1】:

查看此应用的源代码可能会有所帮助。我找到了一个类 AsyncRingtonePlayer (https://android.googlesource.com/platform/packages/apps/DeskClock/+/master/src/com/android/deskclock/AsyncRingtonePlayer.java) 可以完成这项工作。

我猜,诀窍是使用不同的音频流。让我们看看 startAlarm 方法:

 private void startAlarm(MediaPlayer player) throws IOException 
        // do not play alarms if stream volume is 0 (typically because ringer mode is silent).
        if (mAudioManager.getStreamVolume(AudioManager.STREAM_ALARM) != 0) 
            if (Utils.isLOrLater()) 
                player.setAudioAttributes(new AudioAttributes.Builder()
                        .setUsage(AudioAttributes.USAGE_ALARM)
                        .setContentType(AudioAttributes.CONTENT_TYPE_SONIFICATION)
                        .build());
            
            player.setAudiostreamType(AudioManager.STREAM_ALARM);
            player.setLooping(true);
            player.prepare();
            mAudioManager.requestAudioFocus(null, AudioManager.STREAM_ALARM,
                    AudioManager.AUDIOFOCUS_GAIN_TRANSIENT);
            player.start();
        
    

播放器对象的音频流类型将设置为AudioManager.STREAM_ALARM。我想这可能是解决方案。我没有测试它,但这是我在源代码中发现的第一件事。

还请注意 AndroidManifest 中列出的一些所需权限:

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="android.permission.WAKE_LOCK"/>
<uses-permission android:name="android.permission.VIBRATE"/>
<uses-permission android:name="android.permission.DISABLE_KEYGUARD" />
<!-- WRITE_SETTINGS is required to record the upcoming alarm prior to L -->
<uses-permission android:name="android.permission.WRITE_SETTINGS" />
<!-- READ_PHONE_STATE is required to determine when a phone call exists prior to M -->
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<!-- READ_EXTERNAL_STORAGE is required to play custom ringtones from the SD card prior to M -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

(https://android.googlesource.com/platform/packages/apps/DeskClock/+/master/AndroidManifest.xml)

【讨论】:

【参考方案2】:

Android 4.4 不需要删除代码

//context = a valid context
AudioManager am = (AudioManager) context.getSystemService(context.AUDIO_SERVICE);
MediaPlayer mp = new MediaPlayer();
Uri loc = Uri.parse("android.resource://com.example.test/" + R.raw.music);
try 
    mp.setDataSource(context, loc);
    mp.setAudioStreamType(AudioManager.STREAM_ALARM);
    mp.setLooping(true);
    mp.prepare();
 catch (IllegalArgumentException | SecurityException| IllegalStateException | IOException e) 
    e.printStackTrace();

am.requestAudioFocus(null, AudioManager.STREAM_ALARM,AudioManager.AUDIOFOCUS_GAIN_TRANSIENT);
mp.start();

【讨论】:

以上是关于Android,插入耳机后如何将音频路由到扬声器?的主要内容,如果未能解决你的问题,请参考以下文章

如何通过 Apple 设备的内部扬声器路由音频,避免使用耳机?

同时向扬声器和耳机输出音频?

电脑显示未插入扬声器和耳机?

蓝牙耳机配对后,如何将音频路由到蓝牙耳机

Android蓝牙耳机/扬声器音频切换

Android蓝牙耳机/扬声器音频切换