Java文件上传

1、上传文件对表单的要求
a、上传文件必须是表单提交数据不能是超链接
b、表单的method必须是post,不能是get
c、表单的enctype的值 必须是 multipart/from-data
d、在表单中添加file表单字段,即<input type=”file”…/>

 <form action="" method="post" enctype="multipart/form-data">
        用户名:<input type="text" name="username"/><br/>
        文件1:<input type="file" name="file1"/><br/>
        文件2:<input type="file" name="file2"/><br/>
        <input type="submit" value="提交"/>
</form>

2、比对文件上传表单和普通文本表单的区别
文件上传表单的enctype=”multipart/form-data”,表示多部件表单数据;
普通文本表单可以不设置enctype属性:
当method=”post”时,enctype的默认值为application/x-www-form-urlencoded,表示使用url编码正文;
当method=”get”时,enctype的默认值为null,没有正文,所以就不需要enctype了。
普通文本表单的 type=’file’ 与 type=’input’ 效果一样,会将文件名作为input 的值传上去,文件的内容上传不上去
文件上传表单 文件上传的数据在http协议的请求体中

input 上传请求正文的数据格式

—————————–9206301312724
Content-Disposition: form-data; name=”f3”

aaa

file 上传请求正文的数据格式

—————————–9206301312724
Content-Disposition: form-data; name=”f1”; filename=”新建文本文档.txt”
Content-Type: text/plain

文件的内容数据

3、servlet对文件上传数据的处理
request.getParameter(String)方法获取指定的表单字段字符内容,但文件上传表单已经不在是字符内容,而是字节内容,所以失效。
这时可以使用request的getInputStream()方法获取ServletInputStream对象,它是InputStream的子类,这个ServletInputStream对象对应整个表单的正文部分(从第一个分隔线开始,到最后),这说明我们需要的解析流中的数据。

假设上传的文件是一个文本文件,我们将其内容打印在控制台
  //通过流来获取 请求的信息      

ServletInputStream requestDatas =
request.getInputStream();//对应整个表单的正文部分
//将上传的数据信息直接打印在控制台
BufferedReader r = new BufferedReader(new
for(String s=”“;(s=r.readLine())!=null;){
System.out.println(s);
}

如果我们 自己来处理上传文件的数据内容是非常麻烦的,而Apache已经帮我们提供了解析它的工具:commons-fileupload

fileupload简单应用
fileupload的核心类有:DiskFileItemFactory、ServletFileUpload、FileItem。
使用fileupload组件的步骤如下:
1. 创建工厂类DiskFileItemFactory对象:DiskFileItemFactory factory = new DiskFileItemFactory()
2. 使用工厂创建解析器对象:ServletFileUpload fileUpload = new ServletFileUpload(factory)
3. 使用解析器来解析request对象:List list = fileUpload.parseRequest(request)

隆重介绍FileItem类,它才是我们最终要的结果。一个FileItem对象对应一个表单项(表单字段)。一个表单中存在文件字段和普通字段,可以使用FileItem类的isFormField()方法来判断表单字段是否为普通字段,如果不是普通字段,那么就是文件字段了。
String getName():获取文件字段的文件名称;
String getString():获取字段的内容,如果是文件字段,那么获取的是文件内容,当然上传的文件必须是文本文件;
String getFieldName():获取字段名称,例如:<input type=”text” name=”username”/>,返回的是username;
String getContentType():获取上传的文件的类型,例如:text/plain。
int getSize():获取上传文件的大小;
boolean isFormField():判断当前表单字段是否为普通文本字段,如果返回false,说明是文件字段;
InputStream getInputStream():获取上传文件对应的输入流;
void write(File):把上传的文件保存到指定文件中。

文件上传代码

/*
     * 写一个简单的上传示例:
       表单包含一个用户名字段,以及一个文件字段;
       Servlet保存上传的文件到uploads目录,显示用户名,文件名,文件大小,文件类型。
     */
    public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        if( ServletFileUpload.isMultipartContent(request)){
            request.setCharacterEncoding("utf-8");
            //创建共厂
            DiskFileItemFactory factory = new DiskFileItemFactory();
            ServletFileUpload fu = new  ServletFileUpload(factory);
            try {
                List<FileItem> fs = fu.parseRequest(request);//表单项
                for (FileItem fileItem : fs) {
                     if (fileItem.isFormField()) {//普通字段
                         System.out.println(fileItem.getFieldName());//打印字段名
                         System.out.println(fileItem.getContentType());
                         System.out.println(fileItem.getSize());
                         System.out.println(fileItem.getString());
                     }
                     else{
                         String fileName = fileItem.getName();
                         if(fileName == null || fileName.isEmpty()) {
                                continue;
                            }

 //获取上传路径
                         String filePath = request.getServletContext().getRealPath("/uploads");
                         File filePathDir  = new File(filePath,fileName);
                         fileItem.write(filePathDir);
                     }
                }

            } catch (Exception e) {
                e.printStackTrace();
            }
        }

下面是上传文件的几个问题
a、上传文件不要放在项目根目录下,避免用户就可以通过浏览器直接访问上传的文件,这是非常危险的。
假如说用户上传了一个a.jsp文件,然后用户在通过浏览器去访问这个a.jsp文件,那么就会执行a.jsp中的内容,如果在a.jsp中有如下语句:Runtime.getRuntime().exec(“shutdown –s –t 1”);,那么你就会…

b、文件名称问题
由于游览器版本问题,上传文件文件名称可能会有差异
IE6获取的上传文件名称是完整路径,而其他浏览器获取的上传文件名称只是文件名称而已。浏览器差异的问题我们还是需要处理一下的。
    String name = file1FileItem.getName();
    int lastIndex = name.lastIndexOf("\\");//获取最后一个“\”的位置
    if(lastIndex != -1) {//注意,如果不是完整路径,那么就不会有“\”的存在。
    name = name.substring(lastIndex + 1);//获取文件名称
    }
    response.getWriter().print(name);
c、中文乱码问题

上传文件名称中包含中文:
当上传的谁的名称中包含中文时,需要设置编码,commons-fileupload组件为我们提供了两种设置编码的方式:
request.setCharacterEncoding(String):这种方式是我们最为熟悉的方式了;
fileUpload.setHeaderEncdoing(String):这种方式的优先级高与前一种。

上传文件的文件内容包含中文:
通常我们不需关心上传文件的内容,因为我们会把上传文件保存到硬盘上!也就是说,文件原来是什么样子,到服务器这边还是什么样子!
但是如果你有这样的需求,非要在控制台显示上传的文件内容,那么你可以使用fileItem.getString(“utf-8”)来处理编码。

d、上传文件同名问题(文件重命名)

通常我们会把用户上传的文件保存到uploads目录下,但如果用户上传了同名文件呢?这会出现覆盖的现象。处理这一问题的手段是使用UUID生成唯一名称,然后再使用“_”连接文件上传的原始名称。或者我们自己定义一组字符串 比如:系统当前时间+文件名 20170112_a.jpg等

e、一个目录不能存放过多的文件(存放目录打散)

一个目录下不应该存放过多的文件,一般一个目录存放1000个文件就是上限了,如果在多,那么打开目录时就会很“卡”。你可以尝试打印C:\WINDOWS\system32目录,你会感觉到的。
也就是说,我们需要把上传的文件放到不同的目录中。但是也不能为每个上传的文件一个目录,这种方式会导致目录过多。所以我们应该采用某种算法来“打散”!
打散的方法有很多,例如使用日期来打散,每天生成一个目录。也可以使用文件名的首字母来生成目录,相同首字母的文件放到同一目录下。
日期打散算法:如果某一天上传的文件过多,那么也会出现一个目录文件过多的情况;
首字母打散算法:如果文件名是中文的,因为中文过多,所以会导致目录过多的现象。

我们这里使用hash算法来打散:
1. 获取文件名称的hashCode:int hCode = name.hashCode();;
2. 获取hCode的低4位,然后转换成16进制字符;
3. 获取hCode的5~8位,然后转换成16进制字符;
4. 使用这两个16进制的字符生成目录链。例如低4位字符为“5”

这种算法的好处是,在uploads目录下最多生成16个目录,而每个目录下最多再生成16个目录,即256个目录,所有上传的文件都放到这256个目录下。如果每个目录上限为1000个文件,那么一共可以保存256000个文件。

f、上传的单个文件的大小限制

限制上传文件的大小很简单,ServletFileUpload类的setFileSizeMax(long)就可以了。参数就是上传文件的上限字节数,例如servletFileUpload.setFileSizeMax(1024*10)表示上限为10KB。
一旦上传的文件超出了上限,那么就会抛出FileUploadBase.FileSizeLimitExceededException异常。我们可以在Servlet中获取这个异常,然后向页面输出“上传的文件超出限制”。
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
request.setCharacterEncoding(“utf-8”);
DiskFileItemFactory dfif = new DiskFileItemFactory();
ServletFileUpload fileUpload = new ServletFileUpload(dfif);
// 设置上传的单个文件的上限为10KB
fileUpload.setFileSizeMax(1024 * 10);
try {
List list = fileUpload.parseRequest(request);
//获取第二个表单项,因为第一个表单项是username,第二个才是file表单项
FileItem fileItem = list.get(1);
String name = fileItem.getName();//获取文件名称

        // 如果客户端使用的是IE6,那么需要从完整路径中获取文件名称
        int lastIndex = name.lastIndexOf("\\");
        if(lastIndex != -1) {
            name = name.substring(lastIndex + 1);
        }

        // 获取上传文件的保存目录
        String savepath = this.getServletContext().getRealPath("/WEB-INF/uploads");
        String uuid = CommonUtils.uuid();//生成uuid
        String filename = uuid + "_" + name;//新的文件名称为uuid + 下划线 + 原始名称

        int hCode = name.hashCode();//获取文件名的hashCode
        //获取hCode的低4位,并转换成16进制字符串
        String dir1 = Integer.toHexString(hCode & 0xF);
        //获取hCode的低5~8位,并转换成16进制字符串
        String dir2 = Integer.toHexString(hCode >>> 4 & 0xF);
        //与文件保存目录连接成完整路径
        savepath = savepath + "/" + dir1 + "/" + dir2;
        //因为这个路径可能不存在,所以创建成File对象,再创建目录链,确保目录在保存文件之前已经存在
        new File(savepath).mkdirs();

        //创建file对象,下面会把上传文件保存到这个file指定的路径
        //savepath,即上传文件的保存目录
        //filename,文件名称
        File file = new File(savepath, filename);

        // 保存文件
        fileItem.write(file);
    } catch (Exception e) {
        // 判断抛出的异常的类型是否为FileUploadBase.FileSizeLimitExceededException
        // 如果是,说明上传文件时超出了限制。
        if(e instanceof FileUploadBase.FileSizeLimitExceededException) {
            // 在request中保存错误信息
            request.setAttribute("msg", "上传失败!上传的文件超出了10KB!");
            // 转发到index.jsp页面中!在index.jsp页面中需要使用${msg}来显示错误信息
            request.getRequestDispatcher("/index.jsp").forward(request, response);
            return;
        }
        throw new ServletException(e);
    } 
}
g、上传文件的总大小限制

有时我们需要限制一个请求的大小。也就是说这个请求的最大字节数(所有表单项之和)!实现这一功能也很简单,只需要调用ServletFileUpload类的setSizeMax(long)方法即可。
例如fileUpload.setSizeMax(1024 * 10);,显示整个请求的上限为10KB。当请求大小超出10KB时,ServletFileUpload类的parseRequest()方法会抛出FileUploadBase.SizeLimitExceededException异常。

h、缓存大小与临时目录

大家想一想,如果我上传一个蓝光电影,先把电影保存到内存中,然后再通过内存copy到服务器硬盘上,那么你的内存能吃的消么?
所以fileupload组件不可能把文件都保存在内存中,fileupload会判断文件大小是否超出10KB,如果是那么就把文件保存到硬盘上,如果没有超出,那么就保存在内存中。
  10KB是fileupload默认的值,我们可以来设置它。
  当文件保存到硬盘时,fileupload是把文件保存到系统临时目录,当然你也可以去设置临时目录。
DiskFileItemFactory dfif = new DiskFileItemFactory(1024*20, new File(“F:\temp”));
第一个参数:指定缓存大小为20KB,当上传的文件超出了20KB就会保存到临时目录。
第二个参数:临时目录为F:\temp

注意:本文fileupload指的是 ServletFileUpload 该类对象

以上是本人学习传智播客视频整理而出,仅仅作为本人学习笔记

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值