英文:
Android java mediaplayer - onCompletionListener triggered multiple times after onPrepared()
问题
问题摘要。
我的应用程序根据从传感器接收的蓝牙数据触发音频消息。它以200Hz的速率接收数据(每5毫秒一次)。为了在不必等待这些传感器数据的情况下进行测试,我开发了一个模拟器(作为我在Android上运行的程序的一部分),它从文件中读取记录的传感器数据,并以与传感器通常将数据发送到手机的速度相同的速度播放这些数据(此代码使用了大量的多任务处理和调度以确保正确的时机)。
这些音频消息是使用Java的MediaPlayer播放的。有时,基于传入的数据,播放的音频会被中断,使用mediaplayer.stop()进行中断。然后我使用prepare()来准备好在下次数据要求时再次开始播放音频。
问题在于,在prepare()调用之后,有时会出现不正常的onCompletion调用,我不明白它们是如何触发的/如何摆脱它们。
预期和实际结果。
我有一个CMediaPlayer类,如下所示:
public class CMediaPlayer extends MediaPlayer {
// 类的成员变量和构造函数等
// ...
private void makeStartReady(String fileName, MediaPlayer.OnCompletionListener cListen, MediaPlayer.OnPreparedListener pListen) {
try {
setDataSource(fileName);
prepare();
setOnCompletionListener(cListen);
setOnPreparedListener(pListen);
setVolume(1.0f,1.0f);
} catch (Exception e) {
// 异常处理
// ...
}
}
// 其他方法等
// ...
}
onPreparedListener的样子如下:
(MediaPlayer mp) -> {
if (mState.debugWriter != null)
addDebugLine(mState, ((CMediaPlayer)mp).getName() + ".onPrepared");
((CMediaPlayer)mp).startedAt = -1L;
});
使用startCMP()开始音频,使用stopCMP()停止音频时,当我的onCompletionListener如下所示时,一切正常(即,onPrepared监听器被调用,没有其他情况):
(MediaPlayer mp) -> {
addDebugLine(mState, ((CMediaPlayer)mp).getName() + ".onCompletion "); // + ((CMediaPlayer)mp).startedAt);
}
addDebugLine将调试信息写入CSV文件以供程序运行后分析,如下所示:
public static void addDebugLine(WalkState ws, String label) {
Date now = new Date();
try {
ws.debugWriter.append(Common.formattedNow("HH_mm_ss_SSS") + ","
+ (now.getTime() - ws.startTime) + "," + ws.counter.cur
+ "," + (ws.counter.cur-ws.counter.start) * 5
+ "," + label + "\n");
} catch (IOException e) {
// 异常处理
// ...
}
}
当将onCompletion监听器修改为如下时:
(MediaPlayer mp) -> {
addDebugLine(mState, ((CMediaPlayer)mp).getName() + ".onCompletion ") + ((CMediaPlayer)mp).startedAt);
}
(唯一的区别是向传递给addDebugLine的消息附加了一个Long值),则在onPrepared调用后,会多次调用它。调试CSV文件中的输出(使用MediaPlayer名称"dangerStart")如下所示:
09_59_44_176,34515,6866,34330,dangerStart.onCompletion -1
09_59_44_177,34516,6866,34330,dangerStart.onCompletion -1
09_59_44_177,34516,6866,34330,dangerStart.onCompletion -1
每行的第一个字符串是手机时间,第二个字符串(实际上是一个Long值)是自程序启动以来的时间,第三个是传输数据到手机的硬件的计数器。这个计数器的每次增加1代表5毫秒。
失败会以两种不同的方式发生:它可以在stopCMP()调用之后发生(在onPrepared()之后),甚至可以在startCMP()之后发生,即使没有调用stopCMP(),也会在音频正常完成之前很久。这个问题会在startCMP()调用后的很短时间内发生(大约15毫秒),远在音频的实际完成之前(大约持续5.2秒)。
因此,这个输出意味着名为dangerStart的mediaPlayer在非常短的时间内(大约1毫秒)内调用onCompletion 3次。根据媒体播放器状态图(https://developer.android.com/reference/android/media/MediaPlayer#StateDiagram),这不应该发生。
使用这组数据,将Long值添加到addDebugLine消息中会导致描述的效果。对于其他数据集,情况并非如此,多次onCompletion调用可能会在未扩展addDebugLine消息的情况下发生。我有很多数据点,但找不到它们背后的任何规律或原因。
我希望找出这个数据集出了什么问题,然后在其他数据集上测试解决方案,以确认这确实能够解决普遍的问题。
英文:
- Problem summary.
My app triggers audio messages based on BT data that it receives from sensors. It receives the data at 200Hz (once every 5ms). In order to test without having to wait for data from these sensors, I have developed a simulator (part of my program running on android) that reads recorded sensor data from a file and plays them back at the same speed as the sensors would normally send the data to the phone (this code uses a lot of multitasking and scheduling to get the timing right).
The audio messages are played with the java MediaPlayer. Sometimes, the played audio is interrupted based on the incoming data, with a mediaplayer.stop(). I then use prepare() to get it ready to start playing the next time this is required by the data.
The issue is that after the prepare() call, spurious onCompletion calls sometimes occur and I do not understand how they are triggered / how to get rid of them.
- Expected and actual results.
I have a CMediaPlayer class as follows:
public class CMediaPlayer extends MediaPlayer {
private String name;
private Activity act;
private Cue cue;
public Long startedAt;
public CMediaPlayer (Activity act, Cue rc,
String audioFileName, String audioSet,
String audioFileExtension, String mpName,
MediaPlayer.OnCompletionListener cListen,
MediaPlayer.OnPreparedListener pListen) {
super();
if (checkAndroidPermission(act, Manifest.permission.READ_EXTERNAL_STORAGE)) {
if (audioFileExists(act, audioFileName + audioSet)) {
init(act, rc, mpName);
String fullFileName = Environment.getExternalStorageDirectory().getPath() + "/" + PHONE_AUDIO_DIR + "/" +
audioFileName + audioSet + audioFileExtension;
makeStartReady(fullFileName, cListen, pListen);
} else {
Log.e(act.getString(R.string.app_name), audioFileName + audioSet + " file not found under /" + PHONE_AUDIO_DIR + " directory!");
}
} else {
Log.e(act.getString(R.string.app_name), " Missing READ_EXTERNAL_STORAGE permission for this app.");
}
}
private void init(Activity act, Cue rc, String name) {
this.name = name;
this.act = act;
this.cue = rc;
this.startedAt = 0L;
}
private void makeStartReady(String fileName, MediaPlayer.OnCompletionListener cListen, MediaPlayer.OnPreparedListener pListen) {
try {
setDataSource(fileName);
prepare();
setOnCompletionListener(cListen);
setOnPreparedListener(pListen);
setVolume(1.0f,1.0f);
} catch (Exception e) {
Log.e(act.getString(R.string.app_name), fileName + " file not found under /" + PHONE_AUDIO_DIR + " directory!");
e.printStackTrace();
}
}
public void prepare(Activity act, String msg) {
try {
this.prepare();
} catch (Exception e) {
Log.e(act.getString(R.string.app_name), msg + " failed!");
e.printStackTrace();
}
}
public void setOnCompletionListener(OnCompletionListener listener) {
super.setOnCompletionListener(listener);
}
public String getName() {
return name;
}
public void startCMP() {
this.startedAt = this.cue.mState.counter.cur;
cue.mState.callBackDebugLine(name + ".startCMP(" + this.startedAt + ")");
try {
super.start();
} catch (Exception e) {
Log.e(act.getString(R.string.app_name), "cannot start " + name + e.toString());
if (cue != null) cue.mState.callBackDebugLine("cannot start " + name);
}
}
public void stopCMP() {
if (isPlaying()) {
stop();
if (cue != null) cue.mState.callBackDebugLine(name + ".stopCMP(" + this.startedAt + ")");
try {
prepare();
} catch (Exception e) {
Log.e(act.getString(R.string.app_name), name + ".prepareAsync() failed!");
e.printStackTrace();
}
}
}
public void closeCMP() {
cue.mState.callBackDebugLine( name + ".closeCMP()");
reset();
release();
}
}
The onPreparedListener looks like this:
(MediaPlayer mp) -> {
if (mState.debugWriter != null)
addDebugLine(mState, ((CMediaPlayer)mp).getName() + ".onPrepared");
((CMediaPlayer)mp).startedAt = -1L;
});
Starting audio with startCMP() and stopping it with stopCMP() works correctly (ie, the onPrepared listener is called and nothing more) when my onCompletionListener looks like this:
(MediaPlayer mp) -> {
addDebugLine(mState, ((CMediaPlayer)mp).getName() + ".onCompletion "); // + ((CMediaPlayer)mp).startedAt);
},
addDebugLine writes debug information to a csv file for analysis after running the program as follows:
public static void addDebugLine(WalkState ws, String label) {
Date now = new Date();
try {
ws.debugWriter.append(Common.formattedNow("HH_mm_ss_SSS") + ","
+ (now.getTime() - ws.startTime) + "," + ws.counter.cur
+ "," + (ws.counter.cur-ws.counter.start) * 5
+ "," + label + "\n");
} catch (IOException e) {
Log.d(mCtx.getString(R.string.app_name), "exception in addDebugLine");
e.printStackTrace();
}
}
When the onCompletion listener is modified as
(MediaPlayer mp) -> {
addDebugLine(mState, ((CMediaPlayer)mp).getName() + ".onCompletion ") + ((CMediaPlayer)mp).startedAt);
},
(the only difference is that a Long is appended to the message that is passed to addDebugLine)
then it is called multiple times after the onPrepared is called. The output in the debug csv file (with the name of the MediaPlayer equal to "dangerStart") looks like this:
09_59_44_176,34515,6866,34330,dangerStart.onCompletion -1
09_59_44_177,34516,6866,34330,dangerStart.onCompletion -1
09_59_44_177,34516,6866,34330,dangerStart.onCompletion -1
The first string on each line is the phone time, the 2nd string (actually a Long) the time since start of the program, the 3rd is the counter of the hardware that transmits data to the phone. Each increase by 1 on this counter represents 5ms.
The failure happens in two different ways: it can happen after a stopCMP() call (and after onPrepared()), or even just after a startCMP() when no stopCMP() has been called and long before normal completion of the audio. This happens shortly (like 15ms) after the call to startCMP() and long before actual completion of the audio (which lasts approx 5.2s).
So, this output means that the mediaPlayer with name dangerStart calls onCompletion 3x in a very short time period (roughly 1 ms). As per the media player state diagram (https://developer.android.com/reference/android/media/MediaPlayer#StateDiagram), this should not happen.
With this one set of data, adding the Long to the addDebugLine message results in the effect described. With other sets of data, this is not the case and the multiple onCompletion calls may happen without having expanded the addDebugLine message. I have quite a few datapoints but cannot find any rhyme or reason behind them.
I would hope to find what goes wrong with this data set and then to test the solution on the other data sets to confirm this really fixes the problem in general.
专注分享java语言的经验与见解,让所有开发者获益!
评论