`
zyslovely
  • 浏览: 231523 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

文件上传和下载

 
阅读更多
今日学习的主要内容是——文件的上传与下载,下载我们在之前的学习中已经有实现过。主要就是上传文件的功能。上传文件看似简单,但一项成熟的应用是需要考虑到诸多问题的。下面就让我们一起来学习吧!

实现WEB开发中的文件上传功能,需完成如下二步操作:

在WEB页面中添加上传输入项,<input type=“life” name=“”>,使用时注意:

1.          必须要设置input输入项的name属性,否则浏览器将不会发送上传文件的数据。

2.          必须把input项的enctype属值设为multipart/form-data.设置该值后,浏览器在上传文件时,将把文件数据附带在http请求消息体中,并使用MIME协议对上传的文件进行描述,以方便接收方对上传数据进行解析和处理。



如何在Servlet中读取文件上传数据,并保存到服务器本地硬盘中?

     Request对象提供了一个getInputStream()方法,使用这个方法可以获取浏览器提交过来的输入流。但如果浏览器上传多个文件时,我们应该如何区分开来?这是一项复杂的工作。

     为了方便用户处理文件上传数据,Apache开源组织提供了用于处理上面应用的开源组件——Commons-fileupload。这个组件使用简单,性能也比较优异,所以几乎都使用它来实现上传文件的处理功能。Struts的上传文件处理部分也是使用它实现的。

   

Fileupload组件工作流程:


WEB服务器

request

ServletFil

eupLoad

DiskFileItem

Factory

代表普通字段的

FileItem

代表上传文件1

FileItem

代表上传文件2

FileItem

isFileForm

getFieldName

getString

getInputStream

getName

getInputStream

getName



核心API-DiskFileItemFactory:

     DiskFileItemFactory是创建FileItem对象的工厂,这个工厂常用方法:

1.      public DiskFileItemFactory(int sizeThreshold, java.io.File repository),常用的构造函数。

2.      public void setSizeThreshold(int sizeThreshold),设置内存缓冲区的大小,默认值为10K。当上传文件大于缓冲区大小时, fileupload组件将使用临时文件缓存上传文件。

3.      public void setRepository(java.io.File repository),指定临时文件目录,默认值为System.getProperty("java.io.tmpdir")。

       

核心API-ServletFileupLoad:

ServletFileUpload 负责处理上传的文件数据,并将表单中每个输入项封装到一个FileItem对象中。常用方法有:

1.      boolean isMultipartContent(HttpServletRequest request),判断上传表单是否为上传表单类型。

2.      List parseRequest(HttpServletRequest request),解析request对象,并把表单中的每一个输入项包装到一个fileItem 对象中,并返回一个保存了所有FileItem的list集合。

3.      setFileSizeMax(long fileSizeMax),设置上传文件的最大尺寸值。

4.      setSizeMax(long sizeMax),设置上传文件总量的最大值。

5.      setHeaderEncoding(java.lang.String encoding),设置编码格式。如果文件路径中存在中文可能会造成文件路径乱码,用此方法处理可以解决。

6.      setProgressListener(ProgressListener pListener),设置进程监听器,与AWT和Swing的事件处理机制一样。文件上传一点就会触发ProgressListener,这样我们就可以获取文件上传的进度。



上传文件案例:

public class FileuploadServlet extends HttpServlet {



    public void doGet(HttpServletRequest request, HttpServletResponse response)

           throws ServletException, IOException {

       // 创建文件处理工厂,它用于生成FileItem对象。

       DiskFileItemFactory difactory = new DiskFileItemFactory();

       //设置缓存大小,如果上传文件超过缓存大小,将使用临时目录做为缓存。

       difactory.setSizeThreshold(1024 * 1024);



       // 设置处理工厂缓存的临时目录,此目录下的文件需要手动删除。

       String dir = this.getServletContext().getRealPath("/");

       File filedir = new File(dir + "filetemp");

       if (!filedir.exists())

           filedir.mkdir();

       difactory.setRepository(filedir);



       // 设置文件实际保存的目录

       String userdir = dir + "files";

       File fudir = new File(userdir);

       if (!fudir.exists())

           fudir.mkdir();



       // 创建request的解析器,它会将数据封装到FileItem对象中。

       ServletFileUpload sfu = new ServletFileUpload(difactory);

       // 解析保存在request中的数据并返回list集合

       List list = null;

       try {

           list = sfu.parseRequest(request);

       } catch (FileUploadException e) {

           e.printStackTrace();

       }



       // 遍历list集合,取出每一个输入项的FileItem对象,并分别获取数据

       for (Iterator it = list.iterator(); it.hasNext();) {

           FileItem fi = (FileItem) it.next();

           if (fi.isFormField()) {

              System.out.println(fi.getFieldName());

              System.out.println(fi.getString());

           } else {

              //由于客户端向服务器发送的文件是客户端的全路径,在这我们只需要文件名即可

              String filename = fi.getName();

              int index = filename.lastIndexOf("\\");

              if(index != -1)

                  filename = filename.substring(index+1);

              //向服务器写出文件

              InputStream in = fi.getInputStream();

              FileOutputStream fos = new FileOutputStream(fudir + "/" +filename);

              byte[] buf = new byte[1024];

              int len = -1;

              while((len = in.read(buf)) != -1){

                  fos.write(buf, 0, len);

              }

              // 关闭流

              if(in != null){

                  try{

                     in.close();

                  }finally{

                     if(fos!=null)

                         fos.close();

                  }

              }

           }

       }

    }



    public void doPost(HttpServletRequest request, HttpServletResponse response)

           throws ServletException, IOException {

       doGet(request, response);

    }

}



上面的代码只是功能的练习,实际开发中的文件上传需要考虑诸多因素,我们接下来继续学习。

   

JS动态添加文件上传框和按钮的JavaScript代码:

function add(){

           var file = document.createElement("input");

           file.type = "file";

           file.name = "file";

         

           var butt = document.createElement("input");

           butt.type = "button";

           butt.value = "删除";

           butt.onclick = function rem(){

           //必须使用按钮的父节点DIV的父节点来删除自己和自己的父节点DIV。

               this.parentNode.parentNode.removeChild(this.parentNode);

           };

         

           var div = document.createElement("div");

           div.appendChild(file);

           div.appendChild(butt);

         

           var parent = document.getElementById("files");

           parent.appendChild(div);

        }



上传文件的处理细节(1):

1.      中文文件乱码的问题,可以调用两个方法来设置字符编码:servletUpLoader.setHeaderEncoding()或request.setCharacterEncoding()。我们可以在源文件的创建ServletFileUpload对象后边添加如下代码:

sfu.setHeaderEncoding("UTF-8");



2.      临时文件的删除,如果临时文件大于setSizeThreshold设置的缓存大小,Commons-fileupload组件将使用setRepository设置的临时目录来保存上传的文件,上传完成后我们需要手动调用FileItem.delete来删除临时文件。建议不要修改缓存区大小,如果设置缓存为1MB,1000个用户上传文件就需要1000MB内存,服务器会受不了的。我们删除掉setSizeThreshold的代码,并在每次完成一个文件后添加下而的代码:

// 删除临时目录中的文件

fi.delete();



上传文件的处理细节(2):

1.      在上面的代码中,我们将文件的实际保存目录设置在WEB-INF目录之外。这样外部可以直接访问被上传的文件,这会造成安全问题。比如用户上传了一个带有恶意脚本功能的JSP文件,然后从外部访问执行了JSP文件…后果不堪设想。所以我们将源代码中对应位置处修改如下:

// 之所以放在"WEB-INF"目录下是为了防止上传的文件被直接被访问的安全问题

String userdir = dir + "WEB-INF/files";



2.      一个WEB应用会许多不同的用户访问,不同的用户可能会上传相同名称的文件,如果这样可能会造成文件覆盖的情况发生,所以我们必须保证文件名称的唯一性,我编写一个方法来生成唯一性名称的文件名:

/**

     * 生成具有唯一性的UUID文件名称

     * @param fileName

     * @return

     */

    private String uuidName(String fileName){

       UUID uuid = UUID.randomUUID();

       return uuid.toString() + "_" + fileName;

    }

我们将代码“filename = filename.substring(index + 1);”修改为:filename = uuidName(filename.substring(index + 1));



3.      如果一个目录下的文件过多,会极大减慢文件的访问速度。比如一个目录下的文件如果超过1000个,达到1万个呢?恐怖!我们必须编写一个目录结构生成算法,来分散存上传的文件。我们一个方法:

    /**

     * 使用哈希算法生成的文件路径

     * @param dir

     * @param fileName

     * @return

     */

    private String hashPath(String dir, String fileName) {

       int hashCode = fileName.hashCode();

     

       int dir1 = (hashCode >> 4) & 0xf;

       int dir2 = hashCode & 0xf;

     

       String newpath = dir + "/" + dir1 + "/" + dir2 + "/";

       File file = new File(newpath);

       if(!file.exists()){

           file.mkdirs();

       }

     

       return newpath + uuidName(fileName);

    }



上传文件的处理细节(3)

1.      使用ProgressListener显示上传文件进度,在创建ServletFileUpload之后添加如下代码:

// 设置文件上传进度监听器

       sfu.setProgressListener(new ProgressListener() {

           public void update(long pBytesRead, long pContentLength, int pItems) {

              System.out.println("已上传:" + pBytesRead + " 总大小:"

                     + pContentLength);

           }

});

2.      上面的代码会造成频繁的打印,为了使它在上传一定数量后再打印,比如上传10KB后再打印,我们修改上面的代码如下:

// 设置文件上传进度监听器

       sfu.setProgressListener(new ProgressListener() {

           long temp = -1;



           public void update(long pBytesRead, long pContentLength, int pItems) {

              long size = pBytesRead / 1024 * 1024 * 10;

              if(temp == size)

                  return;

              temp = size;

              if(pBytesRead != -1)

                  System.out.println("已上传:" + pBytesRead + " 总大小:"

                     + pContentLength);

              else

                  System.out.println("上传完成!");

           }

});

上面的代码比较经典,好好回味一下。



文件下载:

    WEB应用中实现文件下载的两种方式:

1.         超链接直接指向下载资源

2.         程序实现下载需设置两个响应头:

(1).  设置Content-Type 的值为:application/x-msdownload。Web 服务器需要告诉浏览器其所输出的内容的类型不是普通的文本文件或 HTML 文件,而是一个要保存到本地的下载文件。

(2).  Web 服务器希望浏览器不直接处理相应的实体内容,而是由用户选择将相应的实体内容保存到一个文件中,这需要设置 Content-Disposition 报头。该报头指定了接收程序处理数据内容的方式,在 HTTP 应用中只有 attachment 是标准方式,attachment 表示要求用户干预。在 attachment 后面还可以指定 filename 参数,该参数是服务器建议浏览器将实体内容保存到文件中的文件名称。在设置 Content-Dispostion 之前一定要指定 Content-Type。



     为实现文件下载,首先我们遍历目录下所有文件,Servlet:

import java.io.File;

import java.io.IOException;

import java.util.HashMap;

import javax.servlet.ServletException;

import javax.servlet.http.HttpServlet;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;





public class ListFileServlet extends HttpServlet {

    public void doGet(HttpServletRequest request, HttpServletResponse response)

           throws ServletException, IOException {

       // 获取目录

       String dir = this.getServletContext().getRealPath("/WEB-INF/files");

       HashMap map = new HashMap();

       listFile(new File(dir), map);

       // 将文件列表设置到request的属性中,然后由JSP页面打印列表。

       request.setAttribute("filemap", map);

       request.getRequestDispatcher("/list.jsp").forward(request, response);

    }



    /**

     * 使用递归算法,将所有子目录中的文件添加到列表中

     *

     * @param f

     * @param l

     */

    private void listFile(File f, HashMap map) {

       if (f.isFile()) {

           String path = f.getAbsolutePath().substring(

                  this.getServletContext().getRealPath("/").length());

           String name = f.getName();

           name = name.substring(name.indexOf("_")+1);        

           //BASE64Encoder encoder = new BASE64Encoder();

           map.put(path, name);

       } else {

           File[] files = f.listFiles();

           for (int i = 0; i < files.length; i++) {

              listFile(files[i], map);

           }

       }



    }



    public void doPost(HttpServletRequest request, HttpServletResponse response)

           throws ServletException, IOException {

       doGet(request, response);

    }

}
分享到:
评论

相关推荐

    文件上传和下载文件上传和下载文件上传和下载

    文件上传和下载是Web应用程序中的核心功能之一,无论是用户向服务器提交个人资料、分享文档,还是从服务器获取资源,如软件更新、电子书籍等,都离不开这一操作。在这个过程中,前端与后端的交互以及数据的安全传输...

    java文件上传和下载功能

    在Java编程领域,文件上传和下载是Web应用中常见的功能,尤其在用户交互丰富的网站或系统中。本项目通过一个简单的源码实例,演示了如何实现在JSP(JavaServer Pages)界面上处理文本和图片的上传与下载,涵盖了相关...

    文件上传和下载,代码编写,步骤

    无论是用户头像上传、文档管理还是多媒体资源分享等场景,都离不开文件的上传与下载操作。本文将详细介绍如何在Java Web应用中实现文件的上传和下载功能,并提供具体的代码示例。 #### 二、文件上传的基本原理 ...

    文件上传和下载的常见测试点

    文件上传和下载测试点总结 文件上传和下载是许多应用程序和网站的基本功能之一,在进行测试时需要考虑多个方面的测试点,以下是文件上传和下载的常见测试点: 下载测试点 1. 右键另存为是否可以正确下载文件,...

    Qt5.8用FTP实现文件上传和下载(带进度条)

    在本文中,我们将深入探讨如何使用Qt5.8框架通过FTP协议实现文件的上传和下载功能,同时结合进度条来实时展示操作进度。Qt是一个功能强大的C++库,提供了丰富的图形用户界面(GUI)工具和网络通信接口,使得开发者...

    jsp文件的上传和下载

    在Java Web开发中,JSP(JavaServer Pages)常常用于创建动态网页,配合Servlet处理用户交互,例如文件的上传和下载。本项目专注于解决在JSP中实现文件上传和下载时遇到的一些常见问题,特别是针对中文文件名的处理...

    ssh文件上传和下载

    ssh下实现用户登陆 管理文件 文件的上传与下载

    PHP文件上传和下载

    在PHP中,文件上传和下载是两个非常重要的功能,它们在Web开发中有着广泛的应用,例如用户上传头像、分享文件或下载资源等。本文将详细介绍如何使用PHP实现这两个功能。 首先,我们从文件上传开始。PHP提供了`$_...

    java代码实现文件上传和下载

    在java代码中实现文件的上传和下载,通过页面的file文件上传到java代码段,获取文件的大小和名字

    javaweb简单实现文件上传与下载源代码

    在JavaWeb开发中,文件上传和下载是常见的功能需求,特别是在构建交互性强的Web应用时。本源代码示例提供了一个简单的实现,帮助开发者理解如何处理这些操作。下面将详细解释涉及的技术点。 1. **文件上传** - **...

    SSH实现文件上传和下载

    文件上传和下载是 J2EE 编程中一个古老而重要的话题,SSH 框架提供了一个简捷方便的文件上传下载的方案,通过一些配置和少量的代码就可以完好解决这个问题。 总体实现 上传文件保存到 T_FILE 表中,T_FILE 表结构...

    asp 文件上传和下载

    在ASP文件上传和下载的场景中,开发者通常会利用ASP技术来实现用户通过Web界面上传文件到服务器,以及从服务器下载文件的功能。下面将详细阐述这个过程中的关键知识点。 一、文件上传 1. HTML表单:文件上传的第一...

    ASP实现文件上传和下载

    另外,"上传下载实例"可能是包含示例代码的ASP文件,演示如何使用这两个组件进行实际操作。这些示例对于初学者来说非常宝贵,因为它们展示了如何在实际项目中集成和使用这些组件。同时,"详细说明文档"应提供了关于...

    javaweb文件上传与下载模块源代码

    在Java Web开发中,文件上传和下载是常见的功能需求,特别是在构建交互性强的Web应用程序时。这个"javaweb文件上传与下载模块源代码"提供了一个实现这些功能的基础框架,对于初学者和Java程序员来说,这是一个很好的...

    文件上传和下载 .rar

    3. 断点续传:为了支持大文件下载和网络中断后的继续,服务器需要记录已下载的部分,并允许客户端通过HTTP的Range头请求特定范围的内容。 三、安全性 1. 验证:在接收文件上传前,应验证用户身份,防止恶意用户...

    C#实现的FTP文件上传和下载

    在本文中,我们将深入探讨如何使用C#编程语言来实现FTP(文件传输协议)客户端功能,包括文件的上传和下载。FTP是一种广泛用于在互联网上交换文件的标准协议,而C#提供了一个强大的类库——`System.Net....

    .NET文件上传和下载

    在.NET平台上进行文件上传和下载是常见的Web应用功能,它涉及到客户端与服务器之间的数据交互,以及数据的安全存储和访问控制。在这个项目中,开发者已经创建了一个包含数据库支持、开源实现的示例,不仅展示了文件...

Global site tag (gtag.js) - Google Analytics