目录
1. 文件就保存在本地,可以用 new File()来获取文件
2. 文件保存在FTP服务器上,可以使用ftpClient.retrieveFile(remotePath,outputStream)
项目中经常用到播放一些文件,比如mp3,wav格式的一些文件,下面提供了一种直接拿到音频文件并响应给前端的一种方式,我的音频文件是存在ftp服务器上,而且ftp服务器和应用服务器是部署在一起的,文件要从ftp服务器上获取,这个可以根据实际存放的地方去拿文件。经过测试,前端只需要 <audio>标签接收就可以了,src路径直接填写接口路径,话不多说,直接上图:
1. 文件就保存在本地,可以用 new File()来获取文件
File music = new File("C:\Users\Temp\a.mp3");
附代码:
import io.swagger.annotations.ApiImplicitParam;
import org.apache.catalina.connector.ClientAbortException;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.*;
@RestController
@RequestMapping("/stream")
public class WavApiController {
@Value("${ftpServer_url}")
private String ftpServer_url;
@Value("${ftpServer_audio_url}")
private String ftpServer_audio_url;
@RequestMapping("/mp3")
@ApiImplicitParam(name = "voiceName", value = "音频文件名", paramType = "query", required = true, dataType = "String")
public void wavStream(HttpServletRequest request, HttpServletResponse response, String voiceName){
String database = ftpServer_url+ftpServer_audio_url;//音频文件所在的文件夹路径
// 文件目录(偷懒没有改变量,名称,此处和FTP没关系,就是文件的绝对路径)
// 也就是 File music = new File("C:\Users\Administrator\AppData\Local\Temp\a.mp3");
File music = new File(database+voiceName);
if (!music.exists()){
System.out.println(database+"不存在 文件名为:"+voiceName+"的音频文件!");
}
String range=request.getHeader("Range");
//开始下载位置
long startByte = 0;
//结束下载位置
long endByte = music.length() - 1;
//有range的话
if (range != null && range.contains("bytes=") && range.contains("-")) {
range = range.substring(range.lastIndexOf("=") + 1).trim();
String ranges[] = range.split("-");
try {
//判断range的类型
if (ranges.length == 1) {
//类型一:bytes=-2343
if (range.startsWith("-")) {
endByte = Long.parseLong(ranges[0]);
}
//类型二:bytes=2343-
else if (range.endsWith("-")) {
startByte = Long.parseLong(ranges[0]);
}
}
//类型三:bytes=22-2343
else if (ranges.length == 2) {
startByte = Long.parseLong(ranges[0]);
endByte = Long.parseLong(ranges[1]);
}
} catch (NumberFormatException e) {
startByte = 0;
endByte = music.length() - 1;
}
}
//要下载的长度
long contentLength = endByte - startByte + 1;
//文件名
String fileName = music.getName();
//文件类型
String contentType = request.getServletContext().getMimeType(fileName);
//各种响应头设置
//参考资料:https://www.ibm.com/developerworks/cn/java/joy-down/index.html
//坑爹地方一:看代码
response.setHeader("Accept-Ranges", "bytes");
//坑爹地方二:http状态码要为206
response.setStatus(206);
response.setContentType(contentType);
response.setHeader("Content-Type", contentType);
//这里文件名换你想要的,inline表示浏览器直接实用(我方便测试用的)
//参考资料:http://hw1287789687.iteye.com/blog/2188500
// response.setHeader("Content-Disposition", "inline;filename=test.mp3");
response.setHeader("Content-Length", String.valueOf(contentLength));
//坑爹地方三:Content-Range,格式为
// [要下载的开始位置]-[结束位置]/[文件总大小]
response.setHeader("Content-Range", "bytes " + startByte + "-" + endByte + "/" + music.length());
BufferedOutputStream outputStream = null;
RandomAccessFile randomAccessFile = null;
//已传送数据大小
long transmitted = 0;
try {
randomAccessFile = new RandomAccessFile(music, "r");
outputStream = new BufferedOutputStream(response.getOutputStream());
byte[] buff = new byte[4096];
int len = 0;
randomAccessFile.seek(startByte);
//坑爹地方四:判断是否到了最后不足4096(buff的length)个byte这个逻辑((transmitted + len) <= contentLength)要放前面!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
//不然会会先读取randomAccessFile,造成后面读取位置出错,找了一天才发现问题所在
while ((transmitted + len) <= contentLength && (len = randomAccessFile.read(buff)) != -1) {
outputStream.write(buff, 0, len);
transmitted += len;
//停一下,方便测试,用的时候删了就行了
Thread.sleep(10);
}
//处理不足buff.length部分
if (transmitted < contentLength) {
len = randomAccessFile.read(buff, 0, (int) (contentLength - transmitted));
outputStream.write(buff, 0, len);
transmitted += len;