【发布时间】:2020-06-17 09:26:32
【问题描述】:
我在使用 android MediaPlayer 时遇到问题,我首先在播放之前将整首歌曲缓存到内存中,然后我决定从我用 SparkJava 编写的 api 流式传输歌曲。所以现在它工作正常如果我试图寻找一个加载点,否则它就会停止。并在 API 服务器中生成:
org.eclipse.jetty.io.EofException
at org.eclipse.jetty.io.ChannelEndPoint.flush(ChannelEndPoint.java:286)
at org.eclipse.jetty.io.WriteFlusher.flush(WriteFlusher.java:393)
at org.eclipse.jetty.io.WriteFlusher.completeWrite(WriteFlusher.java:349)
at org.eclipse.jetty.io.ChannelEndPoint$3.run(ChannelEndPoint.java:133)
at org.eclipse.jetty.util.thread.strategy.EatWhatYouKill.runTask(EatWhatYouKill.java:333)
at org.eclipse.jetty.util.thread.strategy.EatWhatYouKill.doProduce(EatWhatYouKill.java:295)
at org.eclipse.jetty.util.thread.strategy.EatWhatYouKill.tryProduce(EatWhatYouKill.java:168)
at org.eclipse.jetty.util.thread.strategy.EatWhatYouKill.run(EatWhatYouKill.java:126)
at org.eclipse.jetty.util.thread.ReservedThreadExecutor$ReservedThread.run(ReservedThreadExecutor.java:366)
at org.eclipse.jetty.util.thread.QueuedThreadPool.runJob(QueuedThreadPool.java:765)
at org.eclipse.jetty.util.thread.QueuedThreadPool$2.run(QueuedThreadPool.java:683)
at java.base/java.lang.Thread.run(Thread.java:830)
Caused by: java.io.IOException: Connection reset by peer
at java.base/sun.nio.ch.SocketDispatcher.write0(Native Method)
at java.base/sun.nio.ch.SocketDispatcher.write(SocketDispatcher.java:54)
at java.base/sun.nio.ch.IOUtil.writeFromNativeBuffer(IOUtil.java:113)
at java.base/sun.nio.ch.IOUtil.write(IOUtil.java:79)
at java.base/sun.nio.ch.IOUtil.write(IOUtil.java:50)
at java.base/sun.nio.ch.SocketChannelImpl.write(SocketChannelImpl.java:484)
at org.eclipse.jetty.io.ChannelEndPoint.flush(ChannelEndPoint.java:264)
这是在 API 部分使用的代码:
public static Object postAudioResponse(Request request, Response response) {
try ( OutputStream os = response.raw().getOutputStream(); BufferedOutputStream bos = new BufferedOutputStream(os))
{
File mp3 = new File("C:\\FTPServer\\" + request.queryParams("dir"));
String range = request.headers("Range");
if (range == null) {
response.status(200);
byte[] bytes = Files.readAllBytes(java.nio.file.Paths.get("C:\\FTPServer\\" + request.queryParams("dir")));
response.header("Content-Type", "audio/mpeg");
response.header("Content-Length", String.valueOf(bytes.length));
System.out.println(response.raw().toString());
HttpServletResponse raw = response.raw();
raw.getOutputStream().write(bytes);
raw.getOutputStream().flush();
raw.getOutputStream().close();
return raw;
}
int[] fromTo = fromTo(mp3, range);
int length = fromTo[1] - fromTo[0] + 1;
response.status(206);
response.raw().setContentType("audio/mpeg");
response.header("Accept-Ranges", "bytes");
response.header("Content-Range", contentRangeByteString(fromTo));
response.header("Content-Length", String.valueOf(length));
final RandomAccessFile raf = new RandomAccessFile(mp3, "r");
raf.seek(fromTo[0]);
writeAudioToOS(length, raf, bos);
raf.close();
bos.flush();
bos.close();
return response.raw();
} catch (IOException e) {
e.printStackTrace();
response.header("Content-Type", "application/json");
return gson.toJson(new StandardResponse(StatusResponse.ERROR, e.toString()));
}
}
这是我的 API HTTP 响应(字符串格式)
HTTP/1.1 206
Date: Wed, 04 Mar 2020 19:39:45 GMT
Content-Type: audio/mpeg
Accept-Ranges: bytes
Content-Range: bytes 4767443-4775635/8897707
Content-Length: 8193
我尝试了很多东西,更改标题,多次验证标题,尝试了 ExoPlayer,我什至检查了 HTTP 部分的 android 源代码,它似乎是正确的。
安卓代码:
mediaPlayer = new MediaPlayer();
mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
mediaPlayer.setDataSource(Api.getSongSource(songName));
mediaPlayer.prepareAsync();
注意:在我发送包含特定范围内内容的新 HTTP 响应之前发生异常。 谢谢。
* UPD *
我在第一个答案中解决了这个问题。
【问题讨论】:
标签: java android http media spark-java