Java Web项目中实现文件下载功能的实例教程
需求:实现一个具有文件下载功能的网页,主要下载压缩包和图片
两种实现方法:
一:通过超链接实现下载
在html网页中,通过超链接链接到要下载的文件的地址
<!doctype html> <html> <head> <meta charset="utf-8"> <title>insert title here</title> </head> <body> <h1>通过链接下载文件</h1> <a href="/day06/download/cors.zip">压缩包</a> <a href="/day06/download/1.png">图片</a> </body> </html>
其中day06/download是文档路径,本实例的程序结构如下:
程序运行后,可以通过单击需要下载文档实现下载
但是这里会出现一个问题,就是单击下载压缩包的时候会弹出下载页面,但是下载图片的时候浏览器就直接打开了图片,没有下载。
这是因为通过超链接下载文件时,如果浏览器可以识别该文件格式,浏览器就会直接打开。只有浏览器不能识别该文件格式的时候,才会实现下载。因此利用第二种方法实现下载功能。
二:通过servlet程序实现下载
通过servlet下载文件的原理是通过servlet读取目标程序,将资源返回客户端。
<!doctype html> <html> <head> <meta charset="utf-8"> <title>insert title here</title> </head> <body> <h1>通过链接下载文件</h1> <a href="/day06/download/cors.zip">压缩包</a> <a href="/day06/download/1.png">图片</a> <h1>通过servlet程序下载文件</h1> <a href="/day06/servletdownload?filename=cors.zip">压缩包</a> <a href="/day06/servletdownload?filename=1.png">图片</a> </body> </html>
其中,/day06/servletdownload 是servlet程序的映射路径
然后新建一个servlet,名称为servletdownload,url映射为/servletdownload
添加代码如下:
package com.lsgjzhuwei.servlet.response; import java.io.fileinputstream; import java.io.fileoutputstream; import java.io.ioexception; import java.io.inputstream; import java.io.outputstream; import javax.servlet.servletexception; import javax.servlet.annotation.webservlet; import javax.servlet.http.httpservlet; import javax.servlet.http.httpservletrequest; import javax.servlet.http.httpservletresponse; /** * servlet implementation class servletdownload */ @webservlet(asyncsupported = true, urlpatterns = { "/servletdownload" }) public class servletdownload extends httpservlet { private static final long serialversionuid = 1l; /** * @see httpservlet#httpservlet() */ public servletdownload() { super(); // todo auto-generated constructor stub } /** * @see httpservlet#doget(httpservletrequest request, httpservletresponse response) */ protected void doget(httpservletrequest request, httpservletresponse response) throws servletexception, ioexception { // todo auto-generated method stub //获得请求文件名 string filename = request.getparameter("filename"); system.out.println(filename); //设置文件mime类型 response.setcontenttype(getservletcontext().getmimetype(filename)); //设置content-disposition response.setheader("content-disposition", "attachment;filename="+filename); //读取目标文件,通过response将目标文件写到客户端 //获取目标文件的绝对路径 string fullfilename = getservletcontext().getrealpath("/download/" + filename); //system.out.println(fullfilename); //读取文件 inputstream in = new fileinputstream(fullfilename); outputstream out = response.getoutputstream(); //写文件 int b; while((b=in.read())!= -1) { out.write(b); } in.close(); out.close(); } /** * @see httpservlet#dopost(httpservletrequest request, httpservletresponse response) */ protected void dopost(httpservletrequest request, httpservletresponse response) throws servletexception, ioexception { // todo auto-generated method stub } }
重启tomcat服务器,即可实现对压缩包和对图片的下载。
三、小技巧:
点击链接来下载文件的方式很简便,后台把文件流输出来,通过浏览器实现下载功能,包括询问位置与文件存放,大多数浏览器会配置一个固定位置,不一定每次都问。
前端就非常简单了,一个<a>标签,href=“后台方法地址”,如果你的需求不能直接用超链接方式,可以在js里写 window.location.href =“后台方法地址”。
这样跳转到后台方法后
string filepath = this.getclass().getclassloader().getresource("").touri().getpath() + "/exportpdf.pdf"; //文件在项目中的路径 file outfile = new file(filepath); string filename = outfile.getname();// 获取文件名称 inputstream fis = new bufferedinputstream(new fileinputstream( filepath)); byte[] buffer = new byte[fis.available()]; fis.read(buffer); //读取文件流 fis.close(); response.reset(); //重置结果集 response.addheader("content-disposition", "attachment;filename=" + new string(filename.replaceall(" ", "").getbytes("utf-8"), "iso8859-1")); //返回头 文件名 response.addheader("content-length", "" + outfile.length()); //返回头 文件大小 response.setcontenttype("application/octet-stream"); //设置数据种类 //获取返回体输出权 outputstream os = new bufferedoutputstream(response.getoutputstream()); os.write(buffer); // 输出文件 os.flush(); os.close();
浏览器会直接识别这种形式的文件输出,弹出对话框。
注意此方法一定要用链接方式调后台,使用ajax和xmlhttprequest方式都是不行的,这样返回的文件流会返回到方法的回调函数中,当然如果你想在js中获取文件,这样也行。
上一篇: Java实现BASE64编码和解码的方法
下一篇: PHP实现的回溯算法示例