`
liuyp2003
  • 浏览: 56491 次
  • 性别: Icon_minigender_1
  • 来自: 广州
最近访客 更多访客>>
社区版块
存档分类
最新评论

基于表单的文件上传

    博客分类:
  • java
阅读更多

如果在表单中使用表单元素 <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);
分享到:
评论

相关推荐

    基于SpringBoot的文件上传系统,前后端分离,单文件上传,多文件上传,大文件上传,断点续传,文件秒传,图片上传

    基于SpringBoot的文件上传系统,前后端分离,单文件上传,多文件上传,大文件上传,断点续传,文件秒传,图片上传 项目经过严格测试,确保可以运行! 采用前后端分离的方式进行开发,实现了几种常用的文件上传功能...

    表单文件上传(更新)

    "表单文件上传(更新)"这个主题涉及到的技术点主要包括单文件上传、多文件上传、大文件上传以及进度条显示,这些都是提升用户体验的关键要素。 首先,**单文件上传**是指用户可以通过网页表单选择一个文件进行上传。...

    基于jsp的文件上传下载

    "基于jsp的文件上传下载"是Web应用程序中的常见需求,涉及到客户端与服务器之间的数据交互。本项目详细阐述了如何利用JSP实现文件的上传和下载功能。 首先,文件上传涉及的主要技术有HTML表单、Servlet和多部分请求...

    基于servlet的文件上传

    本示例基于Servlet和Apache Commons FileUpload库实现了一个简单的文件上传系统。下面将详细介绍这个过程及其涉及到的关键知识点。 首先,`servlet`是Java EE(企业版)的一部分,它是一种用于扩展Web服务器功能的...

    基于bootstrap.3的文件上传效果, 可拖拽上传、预览的HTML5文件上传插件.zip

    标题中的“基于Bootstrap.3的文件上传效果, 可拖拽上传、预览的HTML5文件上传插件.zip”指的是一个使用Bootstrap 3框架和HTML5技术实现的高级文件上传功能。这个插件允许用户通过拖放操作上传文件,并且在上传前可以...

    嵌入式Linux下基于CGI的文件上传下载的实现

    ### 嵌入式Linux下基于CGI的文件上传下载实现 #### 1. 概述 随着嵌入式Linux的深入研究和发展,其在各领域的应用变得越来越广泛。嵌入式Linux是一种针对特定应用场景进行了裁剪和优化的Linux版本,能够适应资源...

    实现文件上传,以及表单提交成功的回调函数

    `jQuery Form Plugin`是基于jQuery的一个插件,它可以方便地支持文件上传,包括多文件上传(如果浏览器支持),并且提供了进度条显示等功能。 - **后端**:后端需要接收并处理上传的文件。在这个例子中,可能使用...

    在一个form表单里同时上传多个文件和文本信息的解决方案

    注意`enctype`属性设置为`multipart/form-data`,这是为了支持文件上传。 #### 2. 错误页面(error.jsp) ```jsp 错误 An Error has occurred in this application. ``` 当上传失败时,会跳转到此页面显示...

    小程序复杂表单含文件上传demo.rar

    本资源“小程序复杂表单含文件上传demo.rar”提供了一个全面展示如何在小程序中实现复杂表单提交功能,包括表单字段验证、错误提示以及多文件上传与个数限制的示例。以下将详细讲解这些关键知识点。 首先,表单是...

    基于web的上传文件案例

    在构建基于Web的文件上传系统时,我们常常面临各种挑战,比如处理大文件上传、实时显示上传进度等。本案例就是一个专门针对这些问题设计的解决方案,它支持大文件(如2GB)的上传,并且利用Ajax技术实现了文件上传的...

    基于html5的文件上传

    "基于HTML5的文件上传"是指利用HTML5的新特性来实现更高效、更友好的文件上传体验,尤其对于大文件上传,它能提供进度条显示,使用户能够直观地了解文件上传的状态。本项目似乎是一个解决方案,包含一个名为`...

    基于文件上传规范的支持多文件上传思路及实现

    本篇文章将详细探讨基于文件上传规范支持多文件上传的思路与实现方法。 首先,我们需要理解文件上传的基本流程。在HTTP协议中,文件上传通常依赖于`POST`请求,通过`multipart/form-data`编码类型来处理包含文件的...

    不用form提交表单,用ajax上传文件

    通过以上步骤,你就可以实现一个基于Ajax的文件上传功能,无需form提交,提高了用户体验。在实际项目中,还需要考虑安全性、性能优化等方面的问题,例如设置文件大小限制、文件类型检查、防止重复上传等。

    基于struts的文件上传下载源代码

    在本"基于Struts的文件上传下载源代码"中,我们可以深入理解Struts如何处理文件上传和下载操作,这对于初学者来说是一个非常实用的学习资源。 首先,文件上传在Web应用中是常见的功能,它允许用户从本地计算机选择...

    基于WWW文件上传的几种实现

    基于浏览器的文件上传遵循的是RFC 1867中规定的规则,主要通过HTML表单(`&lt;form&gt;`)实现。具体来说,需要设置表单的`enctype`属性为`multipart/form-data`,同时`method`属性设置为`POST`,这样才能确保文件能以正确...

    基于structs+hibernate 文件上传

    在IT行业中,基于Struts2和Hibernate的文件上传是一个常见的需求,特别是在开发Web应用程序时。Struts2是一个Java EE框架,用于构建MVC(模型-视图-控制器)架构的应用程序,而Hibernate则是一个强大的对象关系映射...

    嵌入式Linux下基于CGI的文件上传下载的实现.pdf

    ### 嵌入式Linux下基于CGI的文件上传下载实现 #### 概述与背景 随着嵌入式Linux在各个领域的广泛应用,特别是在资源受限的环境下实现高效、可靠的网络服务变得至关重要。嵌入式Linux是一种经过裁剪和优化的操作...

    基于springMVC的文件上传功能

    本文将深入探讨如何利用SpringMVC实现基于Java的文件上传功能,以及在eclipse开发环境中,如何配置和测试这个功能。 1. **环境配置** 开发环境为eclipse,服务器是Tomcat,Java版本为jdk1.7。首先,你需要确保已...

    基于Struts+Hibernate文件上传

    在文件上传场景中,Struts提供了`ActionForm`类用于封装表单数据,`Action`接口或`ActionSupport`类用于处理业务逻辑,`DispatcherAction`负责调度请求。Struts的配置文件(struts-config.xml)定义了各个Action的...

    基于asp.net 的文件上传和下载

    当用户提交表单时,可以通过控件的`HasFile`属性检查是否有文件上传,然后使用`SaveAs`方法将文件保存到服务器的指定位置。 ```csharp if (FileUpload1.HasFile) { string fileName = Path.GetFileName(FileUpload1...

Global site tag (gtag.js) - Google Analytics