Java文件的上传和下载-小白一看就明白

一、文件上传概述

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

在web页面中添加上传输入项 在servlet中读取上传文件的数据,并保存到服务器硬盘中。

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

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

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

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

3、表单的提交方式要是post

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

Request对象提供了一个getInputStream方法,通过这个方法可以读取到客户端提交过来的数据。但由于用户可能会同时上传多个文件,在servlet端编程直接读取上传数据,并分别解析出相应的文件数据是一项非常麻烦的工作,示例。 为方便用户处理文件上传数据,Apache 开源组织提供了一个用来处理表单文件上传的一个开源组件( Commons-fileupload ),该组件性能优异,并且其API使用极其简单,可以让开发人员轻松实现web文件上传功能,因此在web开发中实现文件上传功能,通常使用Commons-fileupload组件实现。

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

二、fileupload组件工作流程

三、文件上传案例

实现步骤

1、创建DiskFileItemFactory对象,设置缓冲区大小和临时文件目录 2、使用DiskFileItemFactory 对象创建ServletFileUpload对象,并设置上传文件的大小限制。 3、调用ServletFileUpload.parseRequest方法解析request对象,得到一个保存了所有上传内容的List对象。 4、对list进行迭代,每迭代一个FileItem对象,调用其isFormField方法判断是否是上传文件 True 为普通表单字段,则调用getFieldName、getString方法得到字段名和字段值 False 为上传文件,则调用getInputStream方法得到数据输入流,从而读取上传数据。

编码实现文件上传

DiskFileItemFactory 是创建 FileItem 对象的工厂,这个工厂类常用方法: public DiskFileItemFactory(int sizeThreshold, java.io.File repository) 构造函数 public void setSizeThreshold(int sizeThreshold) 设置内存缓冲区的大小,默认值为10K。当上传文件大于缓冲区大小时, fileupload组件将使用临时文件缓存上传文件。 public void setRepository(java.io.File repository) 指定临时文件目录,默认值为System.getProperty("java.io.tmpdir").

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

常用方法有:

boolean isMultipartContent(HttpServletRequest request) 判断上传表单是否为multipart/form-data类型 List parseRequest(HttpServletRequest request) 解析request对象,并把表单中的每一个输入项包装成一个fileItem 对象,并返回一个保存了所有FileItem的list集合。 setFileSizeMax(long fileSizeMax) 设置单个上传文件的最大值 setSizeMax(long sizeMax) 设置上传文件总量的最大值 setHeaderEncoding(java.lang.String encoding) 设置编码格式,解决上传文件名乱码问题 setProgressListener(ProgressListener pListener) 实时监听文件上传状态

FileItem 用来表示文件上传表单中的一个上传文件对象或者普通表单对象

boolean  isFormField() 判断FileItem是一个文件上传对象还是普通表单对象

如果判断是一个普通表单对象

String   getFieldName()  获得普通表单对象的name属性 String  getString(String encoding) 获得普通表单对象的value属性,可以用encoding进行编码设置

如果判断是一个文件上传对象

String  getName() 获得上传文件的文件名(有些浏览器会携带客户端路径)

InputStream getInputStream()  获得上传文件的输入流

delete()  在关闭FileItem输入流后,删除临时文件

多个文件上传的javascript编码

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

如: this.parentNode.parentNode.removeChild(this.parentNode);

	function add(){
		// 在div中添加上传输入项
		document.getElementById("uploaddiv").innerHTML += "<div><input type='file' name='upload' /><input type='button' value='删除' onclick='del(this);' /></div>";
	}

	function del(obj){
		// 传入obj 就是你点击 按钮对象
		var div = obj.parentNode; //获得要删除的div
		div.parentNode.removeChild(div);
	}

四、上传文件常见问题

文件存放位置

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

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

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

五、上传文件的进度监控

ProgressListener显示上传进度

ProgressListener progressListener = new ProgressListener() {
	public void update(long pBytesRead, long pContentLength, int pItems) {

	System.out.println("到现在为止,  " + pBytesRead + " 字节已上传,总大小为 "
			  + pContentLength);
	}
};
upload.setProgressListener(progressListener);

以KB为单位显示上传进度

long temp = -1;   //temp注意设置为类变量
long ctemp = pBytesRead /1024; 
if (mBytes == ctemp)  
	return; 
temp = mBytes; 

六、文件下载

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

方式一 超链接直接指向下载资源

方式二 程序实现下载需设置两个响应头:

设置Content-Type 的值为:下载文件对应MIME 类型、 Web 服务器希望浏览器不直接处理相应的实体内容,而是由用户选择将相应的实体内容保存到一个文件中,这需要设置 Content-Disposition 报头。 在设置 Content-Dispostion 之前一定要指定 Content-Type.

下载案例

遍历指定目录下的所有文件显示给用户,并允许用户完成下载(树的遍历)

中文附件名乱码处理

通过 USER-AGENT 判断浏览器类型

IE(包含MSIE):采用URL编码     URLEncoder.encode

FF(包含Mozilla):采用Base64编码     JavaMail的MimeUtility 工具类     BASE64Encoder对象encode

        public static String base64EncodeFileName(String fileName) {
		BASE64Encoder base64Encoder = new BASE64Encoder();
		try {
			return "=?UTF-8?B?"
					+ new String(base64Encoder.encode(fileName
							.getBytes("UTF-8"))) + "?=";
		} catch (UnsupportedEncodingException e) {
			e.printStackTrace();
			throw new RuntimeException(e);
		}
	}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

纵然间

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值