今天刚刚学到使用服务器端编码的方式实现文件的下载,注意事项还是蛮多的,感觉值得分享一下。在服务器端实现文件下载我踩过以下坑:
1.按照文件拷贝的思路书写以下代码
package com.auicyh.content;
import java.io.FileInputStream;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* *
*<p>Title: DownloadServlet</p>
*<p>Description: </p>
* <p>Company: </p>
* @author yindj
* @date 2018年5月10日下午11:29:31
*/
public class DownloadServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//1.获取要下载的文件的名称
String filename = request.getParameter("filename");
//2.获取要下载的文件的绝对路径
String path = this.getServletContext().getRealPath("download/"+ filename);
//3.获取字节流对象
FileInputStream in = new FileInputStream(path);
//4.获得输出流---通过response获得的输出流 用于向客户端写内容
ServletOutputStream out = response.getOutputStream();
//5.经典文件拷贝代码
int len = 0;
byte[] buffer = new byte[1024];
while((len = in.read(buffer))!=-1){
out.write(buffer, 0, len);
}
//6.关闭流
in.close();
out.close();
}
public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
}
}
结果发现有些文件被浏览器直接解析,并不是下载(比如.jpg文件)
![](https://i-blog.csdnimg.cn/blog_migrate/b1680f05379b017b6f400eb45df07f2c.png)
![](https://i-blog.csdnimg.cn/blog_migrate/e2fcf64ea1207225dd3ad6b3f788d987.png)
查阅资料之后,发现在下载文件之前是要为下载的文件设置MIME类型并且告诉浏览器文件被打开的方式
//要下载的这个文件的类型-----客户端通过文件的MIME类型去区分类型
response.setContentType(this.getServletContext().getMimeType(filename));
//告诉客户端该文件不是直接解析 而是以附件形式打开(下载)
response.setHeader("Content-Disposition", "attachment;filename="+filename);
但是作为一名中国的程序猿始终想在页面上显示点中文,但是当你点击中文的文件时,天刹的Eclipse有意和中国人作对会报以下异常
![](https://i-blog.csdnimg.cn/blog_migrate/36ed0c19d1856e0c748bb3b895af85e4.png)
这是为嘛呢?
2.解决中文的乱码问题
为什么会出现乱码呢?是因为
request.getParameter("filename")获取的文件默认是ISO8859-1,而中文的编码格式是UTF-8。好了,接下来知道该怎么做了吧。怎么做?(我抽颗烟想想...)有了,转码呗。于是有了这句话
//解决获得中文参数的乱码
filename = new String(filename.getBytes("ISO8859-1"),"UTF-8");//美女.jpg
到这里就可以下载了吧?(你想错了,共产党人是没有钱的)中国程序猿的命好苦呀,折腾了这个点了还不行???现在你会发现下载的文件被浏览器解析之后只有扩展名没有文件名(上个图让朕瞧瞧...)
这又是什么梗?
![](https://i-blog.csdnimg.cn/blog_migrate/6c5b538d08618af56716ae269fb92b7a.png)
3.解决文件名字不能解析的问题
下面请听老衲为你慢慢道来,出现这种情况就涉及到浏览器的兼容性问题了,不同的浏览器有不同的解码方式,为了让浏览器解析正确,我们需要为不同的浏览器设置不同的编码方式。那么怎样才能知道用户用的是什么样的浏览器呢?(
相信聪明机制的你应该也想到了。)我们在接收浏览器的请求时通过request可以得到用户的浏览器信息
![](https://i-blog.csdnimg.cn/blog_migrate/5adac3e4266d62194c4b53c9b585d0f8.png)
通过下面的代码为不同的浏览器设置不同的编码方式
if (agent.contains("MSIE")) {
// IE浏览器
filename = URLEncoder.encode(filename, "utf-8");
filename = filename.replace("+", " ");
} else if (agent.contains("Firefox")) {
// 火狐浏览器
BASE64Encoder base64Encoder = new BASE64Encoder();
filename = "=?utf-8?B?"
+ base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
} else {
// 其它浏览器
filename = URLEncoder.encode(filename, "utf-8");
}
到此我们可以愉快的玩耍了。
4.完整代码如下
package com.auicyh.content;
import java.io.FileInputStream;
import java.io.IOException;
import java.net.URLEncoder;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import sun.misc.BASE64Encoder;
/**
* *
* <p>
* Title: DownloadServlet
* </p>
* <p>
* Description:
* </p>
* <p>
* Company:
* </p>
*
* @author yindj
* @date 2018年5月10日下午11:29:31
*/
public class DownloadServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 1.获取要下载的文件的名称
String filename = request.getParameter("filename");
// 2.解决获得中文参数的乱码----下节课讲
filename = new String(filename.getBytes("ISO8859-1"), "UTF-8");// 美女.jpg
// 3.获得请求头中的User-Agent
String agent = request.getHeader("User-Agent");
// 4.根据不同浏览器进行不同的编码
String filenameEncoder = "";
if (agent.contains("MSIE")) {
// IE浏览器
filenameEncoder = URLEncoder.encode(filename, "utf-8");
filenameEncoder = filenameEncoder.replace("+", " ");
} else if (agent.contains("Firefox")) {
// 火狐浏览器
BASE64Encoder base64Encoder = new BASE64Encoder();
filenameEncoder = "=?utf-8?B?" + base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
} else {
// 其它浏览器
filenameEncoder = URLEncoder.encode(filename, "utf-8");
}
// 5.要下载的这个文件的类型-----客户端通过文件的MIME类型去区分类型
response.setContentType(this.getServletContext().getMimeType(filename));
// 6.告诉客户端该文件不是直接解析 而是以附件形式打开(下载)
response.setHeader("Content-Disposition", "attachment;filename=" + filenameEncoder);
// 7.获取要下载的文件的绝对路径
String path = this.getServletContext().getRealPath("download/" + filename);
// 8.获取字节流对象
FileInputStream in = new FileInputStream(path);
// 9.获得输出流---通过response获得的输出流 用于向客户端写内容
ServletOutputStream out = response.getOutputStream();
// 10.经典文件拷贝代码
int len = 0;
byte[] buffer = new byte[1024];
while ((len = in.read(buffer)) != -1) {
out.write(buffer, 0, len);
}
// 11.关闭流
in.close();
out.close();
}
public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
}
}
web.xml文件
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5">
<display-name>WEB14</display-name>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
<welcome-file>index.htm</welcome-file>
<welcome-file>index.jsp</welcome-file>
<welcome-file>default.html</welcome-file>
<welcome-file>default.htm</welcome-file>
<welcome-file>default.jsp</welcome-file>
</welcome-file-list>
<servlet>
<description></description>
<display-name>ByteServlet</display-name>
<servlet-name>ByteServlet</servlet-name>
<servlet-class>com.auicyh.content.ByteServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>ByteServlet</servlet-name>
<url-pattern>/byte</url-pattern>
</servlet-mapping>
<servlet>
<description></description>
<display-name>RefreshServlet</display-name>
<servlet-name>RefreshServlet</servlet-name>
<servlet-class>com.auicyh.heard.RefreshServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>RefreshServlet</servlet-name>
<url-pattern>/refresh</url-pattern>
</servlet-mapping>
<servlet>
<description></description>
<display-name>Servlet1</display-name>
<servlet-name>Servlet1</servlet-name>
<servlet-class>com.auicyh.heard.Servlet1</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>Servlet1</servlet-name>
<url-pattern>/servlet1</url-pattern>
</servlet-mapping>
<servlet>
<description></description>
<display-name>Servlet2</display-name>
<servlet-name>Servlet2</servlet-name>
<servlet-class>com.auicyh.heard.Servlet2</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>Servlet2</servlet-name>
<url-pattern>/servlet2</url-pattern>
</servlet-mapping>
<servlet>
<description></description>
<display-name>DownloadServlet</display-name>
<servlet-name>DownloadServlet</servlet-name>
<servlet-class>com.auicyh.content.DownloadServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>DownloadServlet</servlet-name>
<url-pattern>/downloadServlet</url-pattern>
</servlet-mapping>
</web-app>
download.html文件
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
<h1>使用a标签直接指向服务器上的资源</h1>
<a href="/WEB14/download/a.flv">a.flv</a><br>
<a href="/WEB14/download/a.jpg">a.jpg</a><br>
<a href="/WEB14/download/a.mp3">a.mp3</a><br>
<a href="/WEB14/download/a.mp4">a.mp4</a><br>
<a href="/WEB14/download/a.txt">a.txt</a><br>
<a href="/WEB14/download/a.zip">a.zip</a><br>
<h1>使用服务器端编码的方式实现文件下载</h1>
<a href="/WEB14/downloadServlet?filename=a.flv">a.flv</a><br>
<a href="/WEB14/downloadServlet?filename=a.jpg">a.jpg</a><br>
<a href="/WEB14/downloadServlet?filename=a.mp3">a.mp3</a><br>
<a href="/WEB14/downloadServlet?filename=a.mp4">a.mp4</a><br>
<a href="/WEB14/downloadServlet?filename=a.txt">a.txt</a><br>
<a href="/WEB14/downloadServlet?filename=a.zip">a.zip</a><br>
<a href="/WEB14/downloadServlet?filename=美女.jpg">美女.jpg</a><br>
</body>
</html>