Java文件上传与下载
ninehua 2024-11-17 19:49 45 浏览
1 文件上传
1.1 文件上传入门
1.1.1 实现文件上传条件
1)表单的提交方式必须是POST方式。(才有content-type属性)
2)有文件上传表单,表单中有<input type="file"/>的选择文件的标签
3)把表单设置为enctype="multipart/form-data",提交的数据不再是key-value对,而是字节数据
<form action="${pageContext.request.contextPath }/UploadDemo1" method="post" enctype="multipart/form-data"> 请选择文件: <input type="file" name="img"/><br/> <input type="submit" value="上传" /> </form> |
1.1.2 手动解析上传文件
/** * 手动处理上传文件的逻辑 * @author APPle */ public class UploadDemo1 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //得到实体内容数据 InputStream in = request.getInputStream(); BufferedReader br = new BufferedReader(new InputStreamReader(in)); //读取文件的开始符 String startTag = br.readLine(); //读取文件名: Content-Disposition: form-data; name="img"; filename="news.txt" String line = br.readLine(); String fileName = line.substring(line.lastIndexOf("filename=\"")+10, line.lastIndexOf("\"") ); System.out.println("文件名:"+fileName); //跳过2行 br.readLine(); br.readLine(); //读取文件的实际内容 String str = null; BufferedWriter bw = new BufferedWriter(new FileWriter("E:/files/"+fileName)); while((str=br.readLine())!=null){ //读到文件结束符时退出循环 if((startTag+"--").equals(str)){ break; } //把内容写出文件中 bw.write(str); bw.newLine(); bw.flush(); } //关闭 bw.close(); br.close(); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } } |
1.2 工具实现文件上传
1.2.1 commons-fileupload组件概述
commons-fileupload是Apache组织旗下的开源的文件上传的组件。使用非常简单易用。
1.2.2 核心的API
DiskFileItemFactory类: 用于创建上传对象,设置文件缓存区大小,设置文件缓存目录。
ServletFileUpload类: 用于在Servlet程序中实现文件上传
List<FileItem> list = parseRequest(request): 用于解析请求数据,提取和封装文件信息。
FileItem类: 封装一个文件的所有相关的信息javabean。包含文件名称,文件大小,文件类型, 文件数据内容。
1.2.3 fileUpload开发步骤
1)导入commoms-fileuload的jar包
commons-fileupload-1.2.2.jar 核心包
commons-io-2.1.jar 辅助包
2)编写程序
1.3 实现单文件上传
//1.创建DiskFileItemFactory类 /** * 参数一: 表示文件缓存区的大小。如果上传的文件没有超过缓存区大小,则文件不缓存;否则缓存文件,缓存到临时目录。(byte) * 参数二: 表示缓存区的临时目录。 */ DiskFileItemFactory factory = new DiskFileItemFactory(10*1024,new File("e:/temp/")); //2.创建ServletFileUpload类 ServletFileUpload upload = new ServletFileUpload(factory); /** * 设置文件名的编码 */ upload.setHeaderEncoding("utf-8"); //3.解析request数据(把每一个文件封装到FileItem对象中,FileItem放入List中) try { List<FileItem> list = upload.parseRequest(request); //取出第一个上传的文件 FileItem file = list.get(0); //得到文件名(getName()) String fileName = file.getName(); //得到文件大小 long fileSize = file.getSize(); //得到内容类型 String contentType = file.getContentType(); //得到文件数据内容 InputStream in = file.getInputStream(); /** * 4.把文件数据内容存储到服务器端的硬盘中 */ FileUtils.copyInputStreamToFile(in, new File("e:/files/"+fileName)); /** * 5.文件上传完毕,手动清理缓存文件 */ file.delete(); System.out.println("文件名:"+fileName); System.out.println("文件大小:"+fileSize); System.out.println("文件类型:"+contentType); System.out.println("文件数据内容:"+in); } catch (FileUploadException e) { e.printStackTrace(); } |
1.4 实现多文件上传
//1.创建DiskFileItemFactory对象 DiskFileItemFactory factory = new DiskFileItemFactory(10*1024, new File("e:/temp/")); //2.创建ServletFileUpload对象 ServletFileUpload upload = new ServletFileUpload(factory); //3.设置文件编码 upload.setHeaderEncoding("utf-8"); //4.开始解析文件 try { List<FileItem> list = upload.parseRequest(request); if(list!=null){ List<UploadFile> ufList = new ArrayList<UploadFile>(); //遍历多个文件 for(FileItem file: list){ //取出文件相关信息 String fileName = file.getName(); long fileSize = file.getSize(); String contentType = file.getContentType(); //封装到javabean中 UploadFile uf = new UploadFile(); uf.setFileName(fileName); uf.setFileSize(fileSize); uf.setFileType(contentType); //放入list中 ufList.add(uf); //把文件保存到服务器端的硬盘 FileUtils.copyInputStreamToFile(file.getInputStream(), new File("e:/files/"+fileName)); //删除缓存文件 file.delete(); } request.setAttribute("ufList", ufList); request.getRequestDispatcher("/success.jsp").forward(request, response); } } catch (FileUploadException e) { e.printStackTrace(); } |
1.5 动态选择多文件上传
<html> <head> <title>使用组件实现动态多文件上传</title> </head> <body> <form action="${pageContext.request.contextPath }/UploadDemo3" method="post" enctype="multipart/form-data" name="uploadForm"> <table border="1" width="400px"> <tbody> <tr id="1"> <td> 请选择文件: </td> <td> <input type="file" name="file"/><input type="button" value="删除" onclick="delItem(1)"/> </td> </tr> </tbody> <tr> <td colspan="2"><input type="button" value="添加" onclick="addIten()"/></td> </tr> <tr> <td colspan="2"><input type="button" value="上传" onclick="checkSunbit()"/></td> </tr> </table> </form> <script type="text/javascript"> var id = 2; //添加一行 function addIten(){ var trNode = document.createElement("tr"); trNode.setAttribute("id", id); var tdNode1 = document.createElement("td"); tdNode1.innerHTML = "请选择文件:"; var tdNode2 = document.createElement("td"); var input1 = document.createElement("input"); input1.setAttribute("type", "file"); input1.setAttribute("name", "file"); var input2 = document.createElement("input"); input2.setAttribute("type", "button"); input2.setAttribute("value", "删除"); input2.setAttribute("onclick", "delItem("+id+")"); tdNode2.appendChild(input1); tdNode2.appendChild(input2); trNode.appendChild(tdNode1); trNode.appendChild(tdNode2); var tbodyNode = document.getElementsByTagName("tbody")[0]; tbodyNode.appendChild(trNode); id++; } //删除一行(根据tr的id值删除) function delItem(id){ if(id>1){ var trNode = document.getElementById(id); var tbodyNode = document.getElementsByTagName("tbody")[0]; tbodyNode.removeChild(trNode); id--; } } //提交并且检查file属性 function checkSunbit(){ //检查file属性是否全部填上 var fileList = document.getElementsByName("file"); for(var i=0;i<fileList.length;i++){ //如果为选择file,则其value值为空 if(fileList[i].value==null || fileList[i].value==""){ alert("请选择第"+(i+1)+"个文件"); return; } } //提交表单 var form = document.forms['uploadForm']; form.submit(); } </script> </body> </html> |
1.6 文件上传的细节
1.6.1 限制文件类型
//得到文件类型 String contentType = file.getContentType(); System.out.println(contentType); //如果是图片,才可以上传(image/bmp,jepg,jpg,gif) if(!contentType.toLowerCase().matches("image/[a-z]*")){ throw new FileTypeErrorException("文件类型不符合条件!"); } |
1.6.2 限制上传文件大小
ServletFileUpload类:
setFileSizeMax:设置单个文件的最大容量。
setSizeMax : 设置所有文件的最大容量。
//4.解析请求 try { List<FileItem> list = upload.parseRequest(request); if(list!=null){ for(FileItem file : list){ /** * 限制文件类型 */ //得到文件类型 String contentType = file.getContentType(); System.out.println(contentType); //如果是图片,才可以上传(image/bmp,jepg,jpg,gif) if(!contentType.toLowerCase().matches("image/[a-z]*")){ throw new FileTypeErrorException("文件类型不符合条件!"); } //保存文件 FileUtils.copyInputStreamToFile(file.getInputStream(), new File("e:/files/"+file.getName())); //删除缓存文件 file.delete(); } } } catch (FileTypeErrorException e) { //e.printStackTrace(); //处理文件类型错误的异常 request.setAttribute("message", e.getMessage()); request.getRequestDispatcher("/05.upload.jsp").forward(request, response); return; } catch (FileSizeLimitExceededException e) { //e.printStackTrace(); //处理文件超过限制的异常 request.setAttribute("message", "单个文件不能超过1M"); request.getRequestDispatcher("/05.upload.jsp").forward(request, response); return; } catch (SizeLimitExceededException e) { //e.printStackTrace(); //处理文件超过限制的异常 request.setAttribute("message", "所有文件不能超过5M"); request.getRequestDispatcher("/05.upload.jsp").forward(request, response); return; } catch (FileUploadException e) { e.printStackTrace(); } |
1.6.3 查看文件上传的进度
ServletFileUpload类:
setProgressListener(进度监听器): 设置文件上传的监听器
1.6.4 处理文件名重复问题
/** * 解决文件名重复问题: * 1)日期_时间_随机数.jpg * 2)使用UUID算法(在一台PC都是唯一的) */ String fileName = file.getName(); //得到源文件的后缀名 String supfix = fileName.substring(fileName.lastIndexOf(".")); //.jpg //使用UUID算法生成随机名称 fileName = UUID.randomUUID().toString()+supfix; |
1.6.5 把同一个目录的文件打散到不同的目录下
假如所有文件都放在同一个目录下,不方便管理。
1.6.6 得到普通文本控件内容
//判断该FileItem是否是文件还是普通文本空间 if(file.isFormField()){ //普通文本空间(text/password/checkbox/radio/select/texearea) //得到控件内容 //处理一个普通文本 /*String info = file.getString("utf-8"); System.out.println("描述:"+info);*/ //处理多个普通文本 String fieldName = file.getFieldName(); if("info1".equals(fieldName)){ String info1 = file.getString("utf-8"); System.out.println("描述1:"+info1); } if("info2".equals(fieldName)){ String info2 = file.getString("utf-8"); System.out.println("描述2:"+info2); } } |
2 文件下载
2.1 普通文件下载
使用超链接。缺点:1)暴露文件的路径 2)扩展型和安全性不好
2.2 使用servlet程序下载(推荐)
//得到需要下载的文件 String path = this.getServletContext().getRealPath("/upload/9/1/图片1.png"); File file = new File(path); //读取服务器本地的文件 FileInputStream in = new FileInputStream(file); /** * 处理URL编码问题 */ String fileName = file.getName(); //对文件名进行URl编码 fileName = URLEncoder.encode(fileName, "utf-8"); //判断不同浏览器 String userAgent = request.getHeader("user-agent"); String filefix = null; if(userAgent.contains("Trident")){ //IE filefix = "filename="+fileName; }else if(userAgent.contains("Firefox")){ //Firefox filefix = "filename*="+fileName; }else{ filefix = "filename="+fileName; } //告诉浏览器以下载方式打开资源 response.setHeader("Content-Disposition", "attachment;"+filefix); //把本地文件发送给浏览器 byte[] buf = new byte[1024]; int len = 0; while( (len=in.read(buf))!=-1 ){ response.getOutputStream().write(buf, 0, len); } //关闭 in.close(); |
相关推荐
- Origin将普通双Y轴柱状图升级为高级的3D倒影柱状图
-
双Y轴柱状图可以在同一张图表中展示两组具有不同量纲或数量级的数据。3D倒影柱状图是一种通过三维立体及倒影形式展示多个数据的图表,它通过柱子的高度和颜色来表示不同的数据类别和数值大小,以及通过不同平面展...
- Origin 2021 安装教程(附安装包下载)
-
Origin2021软件介绍广泛用于科研、工程和数据分析领域。它支持多种数据格式,包括Excel、ASCII、CSV等,可以方便地导入和分析各种类型的数据。Origin提供了丰富的数据分析工具,包括...
- VIVO origin os系统关闭广告教程(IQOO同理)
-
以下是我使用期间发现的,如有其他未发现的,欢迎大家评论补充。1设置-安全-更多安全设置-广告与隐私-个性化广告推荐关闭2设置-安全-更多安全设置-应用安装-应用推荐关闭3设置-系统管理-全局搜索-首页...
- Origin 2019 安装教程(附安装包下载)
-
Origin2019软件介绍广泛用于科研、工程和数据分析领域。它支持多种数据格式,包括Excel、ASCII、CSV等,可以方便地导入和分析各种类型的数据。Origin提供了丰富的数据分析工具,包括...
- vivo开启Origin OS3系统公测,并且公布首批至第四批可升级名单
-
vivo在本月初发布了OriginOS3.0操作系统,11月25日开启了首批机型的公测招募活动,首批公测机型包括vivo和iQOO,共计14款机型。首批公测招募的机型有:vivoxfold+、v...
- Origin将普通热图升级为高级的3D堆叠热图
-
3D堆叠热图是一种将多个热图组合在一起展示的图表可视化形式。每个热图代表一个数据集,通过比较不同数据集在不同条件(分组)下的颜色深浅变化来挖掘数据的大小差异或者相关性,能够直观地比较多组数据的相对强度...
- 最新最详细的Origin2021安装教程(保姆级)
-
直入正题,下面是Origin2021的安装过程安装包(复制到浏览器打开):%74%2E%63%6E/%41%36%4E%67%6D%54%74%4A安装步骤1、下载得到安装包后,请先解压!一定要先解压...
- vivo和iQOO抓紧更新 OriginOS5末班车 12月最后一波升级
-
进入12月底了,很多老旧机型也已经开始更新各家的新系统了,所以大家一定要注意查看系统版本更新。那么就拿今年国内销量第一的vivo和iQOO来说,OriginOS5都已经推出两个月了,根据之前公布的适...
- Origin将普通点线图转换为高级的3D堆叠点线图
-
点线图用于显示数据随时间或其他连续变量的变化趋势;点线图主要功能是展示数据随时间或连续变量的变化趋势,强调变化幅度,比较多个变量,描述周期性变化,识别异常值,并帮助分析趋势和预测未来趋势。当具有多组数...
- 蓝厂Origin OS 2.0首批适配机型曝光!有没有你的手机型号?
-
2020年11月18日蓝厂新系统OriginOS正式发布,并由后来的X60系列首发,新系统发布后网上就有言论:蓝厂补足了最后一块短板!发布至今大概已经有11月的时间了,网上也是褒贬不一,Origin...
- Origin2018软件安装包以及安装教程
-
安装步骤:1.鼠标右击软件压缩包,选择“解压到Origin2018”2.打开“Origin2018\Origin2018”文件夹,鼠标右击“setup”,选择“以管理员身份运行”3.软件正在准备...
- Origin这样操作,轻松绘制漂亮的雷达图
-
背景介绍雷达图是以从同一点开始的轴上表示的三个或更多个定量变量的二维图表的形式显示多变量数据的图形方法。轴的相对位置和角度通常没有信息。雷达图也称为网络图、蜘蛛图、星图、星图、蜘蛛网、不规则多边形...
- 曝OriginOS新版本在“玩命打磨底层”流畅性比肩鸿蒙
-
日前,vivo正式官宣了2022年vivo开发者大会的举办时间:11月8日-11月9日。OriginOS全新版本届时将正式登场。时隔一年,这次全新OriginOS又将给我们带来什么样的全新体验? ...
- Dify工具使用全场景:1.0.1发布了,升级(功能篇·第8期)
-
上一篇:Dify工具使用全场景:API扩展(功能篇·第7期)一、Dify1.0.1版本升级特性Dify1.0.1版本在稳定性、性能和用户体验方面进行了全面升级,修复了之前版本中的一些已知问题,带来...
- vivo 新系统originOS 升级公测后增加手机内存
-
vivo新系统originOS,通过内存融合、进程优化、应用预载三种技术,一方面调用部分闲置的闪存空间,另一方面低系统对内存的占用,从而可以让原硬件内存增加2-3GB内存使用。vivo新系统ori...