当前位置:编程学习 > JAVA >>

深入体验JavaWeb开发内幕——文件的上传和下载

    关于文件的上传和下载,是我们进行web开发时经常要用到的一个内容,下面我们就来看一看关于文件上传和下载的具体内容吧!

 

一、文件上传概述

 

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

a)       在web页面中添加上传输入项

b)      在servlet中读取上传文件的数据,并保存到本地硬盘中。

 

2、如何在web页面中添加上传输入项?

c)       <input type=“file”>标签用于在web页面中添加文件上传输入项,设置文件上传输入项时须注意:

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

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

 

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

(f)Request对象提供了一个getInputStream方法,通过这个方法可以读取到客户端提交过来的数据。但由于用户可能会同时上传多个文件,在servlet端编程直接读取上传数据,并分别解析出相应的文件数据是一项非常麻烦的工作,示例。

(g)为方便用户处理文件上传数据,Apache 开源组织提供了一个用来处理表单文件上传的一个开源组件( Commons-fileupload ),该组件性能优异,并且其API使用极其简单,可以让开发人员轻松实现web文件上传功能,因此在web开发中实现文件上传功能,通常使用Commons-fileupload组件实现。

4、使用Commons-fileupload组件实现文件上传,需要导入该组件相应的支撑jar包:Commons-fileupload和commons-io。commons-io不属于文件上传组件的开发jar文件,但Commons-fileupload 组件从1.1 版本开始,它工作时需要commons-io包的支持。

 

二、fileupload组件工作流程

     

 

二、           关于文件上传的几个类核心API

 

 1、DiskFileItemFactory

 

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

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

b)       publicvoid setRepository(java.io.File repository) :指定临时文件目录,默认值为System.getProperty("java.io.tmpdir").

c)       publicDiskFileItemFactory(int sizeThreshold, java.io.File repository) :构造函数

 

2、ServletFileUpload

 

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

•       booleanisMultipartContent(HttpServletRequest request) :判断上传表单是否为multipart/form-data类型

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

•       setFileSizeMax(long fileSizeMax):设置上传文件的最大值

•       setSizeMax(long sizeMax):设置上传文件总量的最大值

•       setHeaderEncoding(java.lang.String encoding):设置编码格式

•       setProgressListener(ProgressListener pListener)

     

 

三、           文件上传案例

 

 

  实现步骤:

1、创建DiskFileItemFactory对象,设置缓冲区大小和临时文件目录

2、使用DiskFileItemFactory 对象创建ServletFileUpload对象,并设置上传文件的大小限制。

3、调用ServletFileUpload.parseRequest方法解析request对象,得到一个保存了所有上传内容的List对象。

4、对list进行迭代,每迭代一个FileItem对象,调用其isFormField方法判断是否是上传文件

为普通表单字段,则调用getFieldName、getString方法得到字段名和字段值

 

为上传文件,则调用getInputStream方法得到数据输入流,从而读取上传数据。

 

四、           上传文件的处理细节

1.中文文件乱码问题

a)       文件名中文乱码问题,可调用ServletUpLoader的setHeaderEncoding方法,或者设置request的setCharacterEncoding属性

2、 临时文件的删除问题

a)       由于文件大小超出DiskFileItemFactory.setSizeThreshold方法设置的内存缓冲区的大小时,Commons-fileupload组件将使用临时文件保存上传数据,因此在程序结束时,务必调用FileItem.delete方法删除临时文件。

b)      Delete方法的调用必须位于流关闭之后,否则会出现文件占用,而导致删除失败的情况。

3、 文件存放位置

a)       为保证服务器安全,上传文件应保存在应用程序的WEB-INF目录下,或者不受WEB服务器管理的目录。

b)      为防止多用户上传相同文件名的文件,而导致文件覆盖的情况发生,文件上传程序应保证上传文件具有唯一文件名。

c)       为防止单个目录下文件过多,影响文件读写速度,处理上传文件的程序应根据可能的文件上传总量,选择合适的目录结构生成算法,将上传文件分散存储。

 

4、关于文件上传速度

(1)ProgressListener显示上传进度

 

      ProgressListenerprogressListener = new ProgressListener() {

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

 

       System.out.println("到现在为止,  " + pBytesRead + " 字节已上传,总大小为 "

                       + pContentLength);

       }

};

upload.setProgressListener(progressListener);

 

   (2)以KB为单位显示上传进度

   long temp = -1;   //temp注意设置为类变量

long ctemp =pBytesRead /1024;

if (mBytes ==ctemp)

       return;

temp = mBytes;

 

五、多个文件上传的javascript编码

   技巧:

•       每次动态增加一个文件上传输入框,都把它和删除按纽放置在一个单独的div中,并对删除按纽的onclick事件进行响应,使之删除删除按纽所在的div。

•       如:

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

<html>

<head>

<script>

    function upload(){

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

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

      varfileInput =document.createElement("input");

       fileInput.type="file";

       fileInput.name="files";

     

       vardeleteBtn=document.createElement("input");

      deleteBtn.type="button";

       deleteBtn.value="删除";

     

       deleteBtn.onclick=function(){

           div.parentNode.removeChild(div);

       }

  

补充:Web开发 , 其他 ,
CopyRight © 2012 站长网 编程知识问答 www.zzzyk.com All Rights Reserved
部份技术文章来自网络,