文件上传下载

package cn.itcast.web.servlet;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.List;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
import org.apache.commons.io.IOUtils;

import cn.itcast.utils.FileUploadUtils;

public class Upload4Servlet extends HttpServlet {

    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        
        response.setContentType("text/html;charset=utf-8");

        // 1.创建 DiskFileItemFactory
        File file = new File(this.getServletContext().getRealPath("/temp"));// 获取temp目录部署到tomcat后的绝对磁盘路径
        DiskFileItemFactory factory = new DiskFileItemFactory(1024 * 100, file); //此处使用默认构造即可

        // 2.创建ServletFileUpload
        ServletFileUpload upload = new ServletFileUpload(factory);
        boolean flag = upload.isMultipartContent(request); // 用于判断是否是上传操作.
        if (flag) {
            // 解决上传文件名称中文乱码
            upload.setHeaderEncoding("utf-8");

            // 设置上传文件大小
            // upload.setSizeMax(1024 * 1024 * 10);// 总大小为10m

            try {
                List<FileItem> items = upload.parseRequest(request);// 解析request,得到所有的上传项FileItem

                // 3.得到所有上传项
                for (FileItem item : items) {
                    if (!item.isFormField()) {
                        // 上传组件

                        String name = item.getName(); // 上传文件名称
                        // 得到上传文件真实名称
                        String filename = FileUploadUtils.getRealName(name);

                        // 得到随机名称
                        String uuidname = FileUploadUtils
                                .getUUIDFileName(filename);

                        // 得到随机目录
                        String randomDirectory = FileUploadUtils
                                .getRandomDirectory(filename);
                        // 注意:随机目录可能不存在,需要创建.
                        String parentPath = this.getServletContext()
                                .getRealPath("/upload");  //这里是发不到服务器中的绝大磁盘路径,而不是工程路径
                        File rd = new File(parentPath, randomDirectory);

                        if (!rd.exists()) {
                            rd.mkdirs();
                        }

                        IOUtils.copy(item.getInputStream(),
                                new FileOutputStream(new File(rd, uuidname)));
                        // 删除临时文件
                        item.delete();
                    }
                }

            } catch (FileUploadException e) {
                // e.printStackTrace();
                response.getWriter().write(e.getMessage());
                return;
            }
        } else {
            response.getWriter().write("不是上传操作");
            return;
        }
    }

    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        doGet(request, response);
    }
    
}
 

 

<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>


<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>

<title>多文件上传</title>

<script type="text/javascript">
    function addFile() {
        var div = document.getElementById("content");

        div.innerHTML += "<div><input type='file' name='f'><input type='button' value='remove file' οnclick='removeFile(this)'></div>";
    }

    function removeFile(btn) {

        document.getElementById("content").removeChild(btn.parentNode);

    }
</script>

</head>

<body>

    <input type="button" value="add File" οnclick="addFile();">
    <br>
    <br>
    <form action="${pageContext.request.contextPath}/upload4" method="post"
        encType="multipart/form-data">
        <input type="file" name="f"><br>
        <div id="content"></div>
        <input type="submit" value="上传">
    </form>
</body>
</html>

 

 

文件上传与下载

上传

文件上传概述

  1. 实现web开发中的文件上传功能,需完成如下二步操作:
    • 在web页面中添加上传输入项
    • 在servlet中读取上传文件的数据,并保存到服务器硬盘中。
  2. 如何在web页面中添加上传输入项?
    • <input type=“file”>标签用于在web页面中添加文件上传输入项,设置文件上传输入项时须注意:
    • 1、必须要设置input输入项的name属性,否则浏览器将不会发送上传文件的数据。
    • 2、必须把form的enctype属值设为multipart/form-data.设置该值后,浏览器在上传文件时,将把文件数据附带在http请求消息体中,并使用MIME协议对上传的文件进行描述,以方便接收方对上传数据进行解析和处理。
    • 3、表单的提交方式要是post

 

  1. 如何在Servlet中读取文件上传数据,并保存到本地硬盘中?
    • Request对象提供了一个getInputStream方法,通过这个方法可以读取到客户端提交过来的数据。但由于用户可能会同时上传多个文件,在servlet端编程直接读取上传数据,并分别解析出相应的文件数据是一项非常麻烦的工作,示例。
    • 为方便用户处理文件上传数据,Apache 开源组织提供了一个用来处理表单文件上传的一个开源组件( Commons-fileupload ),该组件性能优异,并且其API使用极其简单,可以让开发人员轻松实现web文件上传功能,因此在web开发中实现文件上传功能,通常使用Commons-fileupload组件实现。
  2. 使用Commons-fileupload组件实现文件上传,需要导入该组件相应的支撑jar包:Commons-fileupload和commons-io。commons-io 不属于文件上传组件的开发jar文件,但Commons-fileupload 组件从1.1 版本开始,它工作时需要commons-io包的支持。

文件上传步骤

  1. 实现步骤

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

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

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

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

      • True 为普通表单字段,则调用getFieldName、getString方法得到字段名和字段值
      • False 为上传文件,则调用getInputStream方法得到数据输入流,从而读取上传数据。

FileUpload上传操作核心API

1、DiskFileItemFactory 磁盘文件项工厂类

public DiskFileItemFactory(int sizeThreshold, java.io.File repository)  构造工厂时,指定内存缓冲区大小和临时文件存放位置

 

public void setSizeThreshold(int sizeThreshold) 设置内存缓冲区大小,默认10 K

 

public void setRepository(java.io.File repository)设置临时文件存放位置,默认System.getProperty("java.io.tmpdir").

 

内存缓冲区: 上传文件时,上传文件的内容优先保存在内存缓冲区中,当上传文件大小超过缓冲区大小,就会在服务器端产生临时文件

临时文件存放位置: 保存超过了内存缓冲区大小上传文件而产生临时文件

* 产生临时文件可以通过 FileItem的delete方法删除

 

2、ServletFileUpload 文件上传核心类

static boolean  isMultipartContent(javax.servlet.http.HttpServletRequest request)  判断request的编码方式是否为multipart/form-data

 

java.util.List parseRequest(javax.servlet.http.HttpServletRequest request) 解析request,将请求体每个部分封装FileItem对象,返回List<FileItem>

 

void setFileSizeMax(long fileSizeMax) 设置单个文件上传大小

void  setSizeMax(long sizeMax) 设置总文件上传大小

 

void setHeaderEncoding(java.lang.String encoding)  设置编码集 解决上传文件名乱码 *****

 

 

3、FileItem 表示文件上传表单中 每个数据部分

boolean isFormField()  判断该数据项是否为文件上传项,true 不是文件上传 false 是文件上传

if(fileItem.isFormField()){

   // 不是上传项

   java.lang.String getFieldName()  获得普通表单项name属性

   java.lang.String getString() / /java.lang.String getString(java.lang.String encoding) 获得普通表单项value属性 传入编码集用来解决输入value乱码

}else{

   // 是上传项

   java.lang.String getName() 获得上传文件名 (注意IE6存在路径)

   java.io.InputStream getInputStream() 获得上传文件内容输入流

   // 上传文件

   void delete()  删除临时文件(删除时,必须要管理输入输出流)

}

 

注意事项:因为文件上传表单采用编码方式multipart/form-data 与传统url编码不同,所有getParameter 方法不能使用 setCharacterEncoding 无法解决输入项乱码问题

 

JavaScript的多文件上传表单

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

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

 

上传文件存在的问题

  • 上传文件后,在服务器端保存位置

第一类存放位置:直接存放WebRoot目录下 和 除WEB-INF META-INF的其它子目录下  例如: WebRoot/upload

* 客户端可以直接在浏览器上通过url访问位置(资料无需通过权限控制,而可以直接访问) ---- 对上传资源安全性要求不高、或者资源需要用户直接可见

* 例如:购物商城商品图片

 

第二类存放位置:放入WEB-INF及其子目录 或者 不受tomcat服务器管理目录 例如: WebRoot/WEB-INF/upload 、c:\ 、d:\abc

* 客户端无法通过URL直接访问,必须由服务器内部程序才能读取 (安全性较高,可以很容易添加权限控制)

* 例如:会员制在线视频

 

  • 上传文件在同一个目录重名问题

如果文件重名,后上传文件就会覆盖先上传文件

 

文件名 UUID

filename = UUID.randomUUID().toString() + "_" + filename;

 

  • 为了防止同一个目录下方上传文件数量过多 ---- 必须采用目录分离算法

1) 按照上传时间进行目录分离 (周、月 )

2) 按照上传用户进行目录分离 ----- 为每个用户建立单独目录

3) 按照固定数量进行目录分离 ------ 假设每个目录只能存放3000个文件 ,每当一个目录存满3000个文件后,创建一个新的目录

4) 按照唯一文件名的hashcode 进行目录分离  

public static String generateRandomDir(String uuidFileName) {

// 获得唯一文件名的hashcode

int hashcode = uuidFileName.hashCode();

// 获得一级目录

int d1 = hashcode & 0xf;       

// 获得二级目录

int d2 = (hashcode >>> 4) & 0xf;

 

return "/" + d2 + "/" + d1;// 共有256目录l

}

 

  • 乱码问题

普通编写项 value属性乱码 ------------- fileItem.getString(编码集);

上传文件项 文件名乱码 --------- fileupload.setHeaderEncoding(编码集);

 

下载

常见文件下载有两种方式

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

如果文件格式浏览器识别,将直接打开文件,显示在浏览器上, 如果文件格式浏览器不识别,将弹出下载窗口

对于浏览器识别格式的文件,通过另存为进行下载

 

客户端访问服务器静态资源文件时,静态资源文件是通过 缺省Servlet返回的,在tomcat配置文件conf/web.xml 找到 --- org.apache.catalina.servlets.DefaultServlet

 

2、编写服务器程序,读取服务器端文件,完成下载

必须设置两个头信息 ,来自MIME协议  Content-Type  Content-Disposition

 

response.setContentType(getServletContext().getMimeType(filename));

response.setHeader("Content-Disposition", "attachment;filename=" + filename); // 以附件形式打开,不管格式浏览器是否识别

处理IE浏览器与Firefox浏览器乱码问题

if (agent.contains("MSIE")) {

// IE浏览器

filename = URLEncoder.encode(filename, "utf-8");

filename = filename.replace("+", " ");

} else if (agent.contains("Firefox")) {

// 火狐浏览器

BASE64Encoder base64Encoder = new BASE64Encoder();

filename = "=?utf-8?B?"

+ base64Encoder.encode(filename.getBytes("utf-8"))

+ "?=";

} else if (agent.contains("Chrome")) {

// google浏览器

filename = URLEncoder.encode(filename, "utf-8");

} else {

// 其它浏览器

filename = URLEncoder.encode(filename, "utf-8");

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值