问题描述:
java 中inputstream流 转成string,再将String转换会inputStream,下载下来的文件,内容损坏,例如下载word文档
使用场景:
底层服务读取到文件内容获得InputStream,因为需要多次接口调用,为了便于数据传递,将InputStream转换为String字符串进行传递,上层服务调用接口,获取String字符串,在转换成InputStream进行IO的读写操作;
问题原因:
如果文件内容是字符型,这种方法没有问题,如果不是字符型的,比如MP3,图片,word文档等,下载下来会无法打开,如上图;
解决办法:
在底层服务InputStream流转换为String前对二进制数据进行base64加密,然后再转为String字符串:
public String inputStream2String(InputStream in) throws IOException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
int len = 0;
byte[] b = new byte[1024];
while ((len = in.read(b, 0, b.length)) != -1) {
baos.write(b, 0, len);
}
byte[] buffer = baos.toByteArray();
//base64加密
return Base64.encodeBase64String(buffer);
}
然后上层服务调用接口获得字符串,再进行base64解密:
Map<String, Object> reMap = gitCodeViewService.gitCodeView(Id, path, version);
String content = (String) reMap.get("content");
//用base64进行解码
byte[] decodeByte = Base64.decodeBase64(content);
//将解码的二进制文件转换为inputStream
InputStream is = new ByteArrayInputStream(decodeByte);
在使用InputStream进行IO的读写操作,下载文件内容就正常了。
下载文件代码:
String content = (String) codeViewMap.get("content");
//用base64进行解码
byte[] decodeByte = Base64.decodeBase64(content);
//将解码的二进制文件转换为inputStream
InputStream is = new ByteArrayInputStream(decodeByte);
String userAgent = request.getHeader("User-Agent");
if (userAgent.contains("MSIE") || userAgent.contains("Trident")) {
//IE浏览器处理
fileName = java.net.URLEncoder.encode(fileName, "UTF-8");
} else {
// 非IE浏览器的处理:
fileName = new String(fileName.getBytes("UTF-8"), "ISO-8859-1");
}
// 设置文件头:最后一个参数是设置下载文件名
response.setHeader("Content-Disposition", "attachment;fileName="+fileName);
// 设置文件ContentType类型,这样设置,会自动判断下载文件类型
response.setContentType("application/octet-stream");
OutputStream os = response.getOutputStream();
// 输入流输出流对拷
int len = 0;
byte[] b = new byte[1024];
while ((len = is.read(b)) > 0) {
os.write(b, 0, len);
}
os.close();
is.close();
springmvc下载文件遇到的坑
java上传文件不难,思路也比较清晰,利用SpringMVC就更简单了。
获取要下载的文件
InputStream in = new FileInputStream(path);
得到输出流
response.getOutputStream()
设置响应头
response.setContentType("application/force-download");
response.setHeader("content-disposition","attachment;filename="+filename);
老套路,拷贝数据
int len = 0;
byte[] b = new byte[1024];
while((len=in.read(b))!=-1){
out.write(b,0,len);
}
但是这次我下载文件的时候,写的没问题,但是就是一直不能下载,每次都是在页面输出了二级制流。
也就是得到一片乱码。找了半天,也没找到哪里错了。
后来把a标签的href换了,添加了一个点击事件,然后用js中的window.location.href就可以下载了。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。