如果在表单中使用表单元素 <input type=“file” />,浏览器在解析表单时,会自动生成一个输入框和一个按钮,输入框可供用户填写本地文件的文件名和路径名,按钮可以让浏览器打开一个文件选择框供用户选择文件:
当表单需要上传文件时,需指定表单 enctype 的值为 multipart/form-data
在 form 元素的语法中,enctype 属性指定将数据发送到服务器时浏览器使用的编码类型。
enctype 属性取值:
application/x-www-form-urlencoded:表单 enctype 属性的默认值。这种编码方案使用有限的字符集,当使用了非字母和数字时,必须用”%HH”代替(H 代表十六进制数字)。对于大容量的二进制数据或包含非 ASCII 字符的文本来说,这种编码不能满足要求。
multipart/form-data:form 设定了enctype=“multipart/form-data”属性后,表示表单以二进制传输数据 .
Commons-fileupload 组件上传的基本原理
FileUpload组件将页面提交的所有元素(普通form表单域,如text和文件域file)都看作一样的FileItem,这样上传页面提交的 request请求也就是一个FileItem的有序组合,FileUpload组件可以解析该request,并返回一个一个的FileItem。而对每一个FileItem,FileUpload组件可以判断出它是普通form表单域还是文件file域,从而根据不同的类型,采取不同的操作--如果是表单域,就读出其值,如果是文件域,就保存文件到服务器硬盘上或者内存中
Commons-fileupload 组件API
ServletFileUpload 负责处理上传的文件数据,并将每部分的数据封装成一到 FileItem 对象中。
DiskFileItemFactory 是创建 FileItem 对象的工厂,在这个工厂类中可以配置内存缓冲区大小和存放临时文件的目录。
ServletFileUpload 在接收上传文件数据时,会将内容保存到内存缓存区中,如果文件内容超过了 DiskFileItemFactory 指定的缓冲区的大小,那么文件将被保存到磁盘上,存储为 DiskFileItemFactory 指定目录中的临时文件。等文件数据都接收完毕后,ServletUpload 在从文件中将数据写入到上传文件目录下的文件中
进行文件上传和下载的具体操作:
首先,加载必要的jar包:“commons-fileupload-1.2.1.jar”,“commons-io-1.4.jar”
步骤:
0. 创建 FileItemFactory 子类 DiskFileItemFactory 的对象
1.为了得到 ServletFileUpload 对象, 先需要得到 FileItemFactory 的一个对象, 然后调用 new ServletFileUpload(FileItemFactory fif); 方法得到 ServletFileUpload 对象
2.为了得到 FileItem 的集合, 先需要得到 ServletFileUpload 对象, 然后调用该对象的 parseRequest() 方法得到 FileItem 的 List
3.从 HttpServletRequest 对象中得到 FileItem 的集合
4. 对上述集合进行遍历:判断是表单域还是文件域
下面是对文件上传相关操作的代码示例。
//1. 验证是否使用 FileUpload 组件
boolean isMultipart = ServletFileUpload.isMultipartContent(request);
//2.1 不使用
if(!isMultipart){
response.getWriter().println("没有文件域");
return;
}
//2.2 使用
//3. 获取 DiskFileItemFactory 对象
DiskFileItemFactory itemFactory = new DiskFileItemFactory();
//4. 获取 ServletFileUpload 对象
ServletFileUpload fileUpload = new ServletFileUpload();
fileUpload.setFileItemFactory(itemFactory);
设置上传文件的最大大小
fileUpload.setFileSizeMax(1000 * 100);
String forwardPage = null;
try {
//5. 调用 ServletFileUpload 的 parseRequest() 方法, 获取 FileItem 的集合
List<FileItem> items = fileUpload.parseRequest(request);
Map<String, String> paramMap = new HashMap<String, String>();
FileItem fileItem = null;
//6. 对 5 获取的集合进行遍历
Iterator<FileItem> it = items.iterator();
while(it.hasNext()){
FileItem item = it.next();
//7. 验证是否为表单域
if(item.isFormField()){
//7.1 是表单域, 把获取到得 表单域的 fieldName 和 value 放在 map 中.
获得表单中该字段的字段名
String fieldName = item.getFieldName();
获得表单中与上面字段名对应的字段值
String value = item.getString("UTF-8");
paramMap.put(fieldName, value);
}
//7.2 不是表单域, 即为文件域. 获取 FileItem 对象, 注意限制条件.
else{
这里获得了与file对应的FileItem对象
fileItem = item;
}
}
//8. 对 7.2 获取的 FileItem 对象进行文件上传操作.
//10.1 获取全路径名
String path = null;
path = this.getServletContext().getRealPath("/photoes");
File photoesDir = new File(path);
if(!photoesDir.exists()){
photoesDir.mkdir();
}
path = path + "/" + paramMap.get("username");
File file = new File(path + ".jpg");
//10.2 上传操作,把用户在页面上传的文件保存到服务器
fileItem.write(file);
上传文件表单的两个准备工作
1. 提交方式改为 POST
2. enctype 属性的取值改为 multipart/form-data
在获取表单值时
1. 不能使用 request.getParameter() 方法, 因为该方法仅能获取字符串, 对二进制无能为力.
2. 使用 fileUpload 组件读取上传文件及表单域字段.
fileUpload 组件的核心思想
fileUpload 组件认为: 所有的表单域(包含文件域和文本域)都是 FileItem 对象.
步骤:
1. 获取 FileItem 对象组成的 List
2. 对 List 进行遍历
3. 判断每一个 FileItem 对象是表单域 还是 文件域, 以进行不同的处理.
详细步骤:
1. 检查表单的 enctype 是否为 multipart/form-data, 以决定是否使用 FileUpload 组件进行解析请求
2. 若表单的 enctype 是否为 multipart/form-data, 则需要获取 封装了所有表单域 的对应的 FileItem 对象的 List: List<FileItem> list = null;
3) 获取 ServletFileUpload 对象, 调用该对象的 public java.util.List parseRequest(javax.servlet.http.HttpServletRequest request)
方法获取封装 FileItem 的List
2) 需要使用 ServletFileUpload 的构造器来创建 ServletFileUpload 对象:
^ 使用任意一个构造器都可以得到 ServletFileUpload 对象, 但都需要先得到 FileItemFactory 对象
* ServletFileUpload upload = new ServletFileUpload();
upload.setFileItemFactory(factory);
* ServletFileUpload upload = new ServletFileUpload(factory);
1) 创建 FileItemFactory 接口是想类对象: DiskFileItemFactory, 直接创建该对象.
3. 可以调用 DiskFileItemFactory 的相关方法来对文件上传进行一些优化:
1) setSizeThreshold(int sizeThreshold):设置使用临时文件夹的大小.
2) setRepository(): 设置临时文件夹
4. 可以调用 ServletUpload 的方法进行文件上传的一些控制
1) setFileSizeMax: 设置单个文件的最大上传值
2) setSizeMax:设置总上传文件的最大值, 以 byte 为单位.
5. 关于中文乱码的问题:
1) 在解析 request 之前: request.setCharacterEncoding("UTF-8"); 可以解决文件域的乱码问题
2) 在解析表单域时需要使用: new String(value.getBytes("iso-8859-1"), "UTF-8"); 解决中文乱码问题.
但是有时候,即使我们按照上面方式设置了,可以还是中文乱码,这个时候问题多半就出在tomcat的配置上了,打开tomcat的service.xml文件,在那里面注意下面部分
<Connector port="8989" protocol="HTTP/1.1"
connectionTimeout="20000"
redirectPort="8443" useBodyEncodingForURI="true"/>
只有加上了红色部分的设置,我们在程序中配置EncodingUrl才能起作用。
下面一段代码是文件下载的相关操作
//1. 设置响应报头 contentType: application/x-msdownload -->
// 告诉浏览器其所输出的内容的类型不是普通的文本文件或 HTML 文件,而是一个要保存到本地的下载文件
//response.setHeader("conent-type", "application/x-msdownload");
response.setContentType("application/x-msdownload");
//2. 设置响应抱头 Content-Disposition: attachment -->
// Web 服务器希望浏览器不直接处理相应的实体内容,而是由用户选择将相应的实体内容保存到一个文件中
response.setHeader("Content-Disposition", "attachment; filename=haha.ppt");
//3. 利用 response.getOutputStream() 进行 io 操作
OutputStream os = response.getOutputStream();
BufferedOutputStream bos = new BufferedOutputStream(os);
//文件的输入流
InputStream is = null;
is = new FileInputStream("E:\\source\\090515\\javaee\\fileupload\\fileUpload.ppt");
BufferedInputStream bis = new BufferedInputStream(is);
int length = 0;
byte [] temp = new byte[1 * 1024 * 10];
while((length = bis.read(temp)) != -1){
bos.write(temp, 0, length);
System.out.println("循环操作中...");
}
bis.close();
bos.close();
但是上面的这种方式无法被迅雷之类的下载工具下载,如果希望被迅雷之类的东西下载,就直接这么写就可以了
request.getRequestDispatcher("/data/fileUpload.ppt")
.forward(request, response);
(提交表单时,程序会自动识别表单哪个是文件,以下自己测试成功的代码,页面表单需要做一点点小的设置 enctype="multipart/form-data" method="post",这两个是必需的)
package com.util;
import java.io.File;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
public class UploadFiles {
public static void uploadFile(HttpServletRequest request, String savePath)
throws Exception {
boolean isMultipart = ServletFileUpload.isMultipartContent(request);
if(!isMultipart){
System.out.println("没有使用FileUpload组件,请在表单(form)中加入“ enctype=\"multipart/form-data\" method=\"post\" ”");
return;
}
DiskFileItemFactory factory = new DiskFileItemFactory();
factory.setSizeThreshold(10240000);// 设置内存缓冲区,超过后写入临时文件
File file = new File(savePath);// 设置临时文件存储位置
if (!file.exists())
file.mkdirs();
factory.setRepository(file);
ServletFileUpload upload = new ServletFileUpload(factory);
upload.setFileSizeMax(10002400000l);// 设置单个文件的最大上传值
upload.setSizeMax(10002400000l);// 设置整个request的最大值
upload.setHeaderEncoding("UTF-8");
List<?> items = upload.parseRequest(request);
FileItem item = null;
String fileName = null;
if (items != null) {
for (int i = 0; i < items.size(); i++) {
item = (FileItem) items.get(i);
fileName = savePath + File.separator + item.getName();
// 保存文件
if (!item.isFormField() && item.getName().length() > 0) {
item.write(new File(fileName));
}
}
}
}
}
分享到:
相关推荐
基于SpringBoot的文件上传系统,前后端分离,单文件上传,多文件上传,大文件上传,断点续传,文件秒传,图片上传 项目经过严格测试,确保可以运行! 采用前后端分离的方式进行开发,实现了几种常用的文件上传功能...
首先,文件上传涉及的主要技术有HTML表单、Servlet和多部分请求。在HTML表单中,我们需要设置`<input type="file">`标签让用户选择要上传的文件。当用户提交表单时,数据将以多部分/FormData格式发送到服务器。JSP...
本示例基于Servlet和Apache Commons FileUpload库实现了一个简单的文件上传系统。下面将详细介绍这个过程及其涉及到的关键知识点。 首先,`servlet`是Java EE(企业版)的一部分,它是一种用于扩展Web服务器功能的...
### 嵌入式Linux下基于CGI的文件上传下载实现 #### 1. 概述 随着嵌入式Linux的深入研究和发展,其在各领域的应用变得越来越广泛。嵌入式Linux是一种针对特定应用场景进行了裁剪和优化的Linux版本,能够适应资源...
在一个Web应用中,经常需要实现用户能够在一个表单中上传多个文件及文本信息的功能。本文将详细介绍如何在基于Tomcat5.0.30与Spring Framework的环境下实现这一功能。 ### 一、背景介绍 在实际开发过程中,很多...
本资源“小程序复杂表单含文件上传demo.rar”提供了一个全面展示如何在小程序中实现复杂表单提交功能,包括表单字段验证、错误提示以及多文件上传与个数限制的示例。以下将详细讲解这些关键知识点。 首先,表单是...
本示例将讲解如何使用ModelForm来实现多文件上传的功能,这对于处理用户上传的多个文件非常有用,比如在上传图片、文档等场景下。 首先,我们需要创建一个Django模型(Model),包含一个或多个FileField或...
"表单文件上传(更新)"这个主题涉及到的技术点主要包括单文件上传、多文件上传、大文件上传以及进度条显示,这些都是提升用户体验的关键要素。 首先,**单文件上传**是指用户可以通过网页表单选择一个文件进行上传。...
标题中的“基于Bootstrap.3的文件上传效果, 可拖拽上传、预览的HTML5文件上传插件.zip”指的是一个使用Bootstrap 3框架和HTML5技术实现的高级文件上传功能。这个插件允许用户通过拖放操作上传文件,并且在上传前可以...
多文件上传允许用户一次性上传多个文件,极大地提高了用户体验和效率。本篇文章将详细探讨基于文件上传规范支持多文件上传的思路与实现方法。 首先,我们需要理解文件上传的基本流程。在HTTP协议中,文件上传通常...
在本文中,我们将深入探讨如何实现一个基于JavaWeb的多文件上传功能,类似于百度文库的体验。这个功能的实现涉及到前端技术如Jquery和Flash,以及后端的Java技术。我们将从以下几个方面来详细讲解这个知识点: 1. *...
### 嵌入式Linux下基于CGI的文件上传下载实现 #### 概述与背景 随着嵌入式Linux在各个领域的广泛应用,特别是在资源受限的环境下实现高效、可靠的网络服务变得至关重要。嵌入式Linux是一种经过裁剪和优化的操作...
`jQuery Form Plugin`是基于jQuery的一个插件,它可以方便地支持文件上传,包括多文件上传(如果浏览器支持),并且提供了进度条显示等功能。 - **后端**:后端需要接收并处理上传的文件。在这个例子中,可能使用...
在本"基于Struts的文件上传下载源代码"中,我们可以深入理解Struts如何处理文件上传和下载操作,这对于初学者来说是一个非常实用的学习资源。 首先,文件上传在Web应用中是常见的功能,它允许用户从本地计算机选择...
"基于Flash的无刷新多文件大文件上传实例"是一个解决方案,它利用Adobe Flash技术来实现不刷新页面的情况下,上传多个大文件的功能。这个技术在早期广泛使用,因为当时HTML5的File API还未普及。以下是对这一实例的...
通过以上步骤,你就可以实现一个基于Ajax的文件上传功能,无需form提交,提高了用户体验。在实际项目中,还需要考虑安全性、性能优化等方面的问题,例如设置文件大小限制、文件类型检查、防止重复上传等。
在SpringMVC的配置文件(如`servlet-context.xml`)中,你需要配置`<mvc:annotation-driven>`元素来启用基于注解的控制器,同时添加`CommonsMultipartResolver`解析器,用于处理多部分请求,例如文件上传。...
3. **多文件上传**:在本实例中,用户可以上传多个文件。这通常通过HTML的`<input type="file" multiple>`属性实现,允许用户选择多个文件进行上传。在服务器端,需要对每个上传的文件进行单独处理。 4. **文件类型...
《基于jQuery的文件上传:jquery-upload-file深度解析》 在Web开发中,文件上传功能是不可或缺的一部分,尤其在用户交互丰富的应用中。jQuery作为一个广泛使用的JavaScript库,为开发者提供了便捷的DOM操作和事件...
"基于HTML5的文件上传"是指利用HTML5的新特性来实现更高效、更友好的文件上传体验,尤其对于大文件上传,它能提供进度条显示,使用户能够直观地了解文件上传的状态。本项目似乎是一个解决方案,包含一个名为`...