commons-fileupload实现文件上传

五迷三道 提交于 2020-02-10 20:33:09

版权声明:本文为 小异常 原创文章,非商用自由转载-保持署名-注明出处,谢谢!
本文网址:https://blog.csdn.net/sun8112133/article/details/79968573

  文件上传在 WEB 开发中应用的十分广泛,是我们每个 web 开发者应知应会的技术!本篇以实例讲解如何使用 commons-fileupload 工具包实现文件上传,并带你领略 commons-fileupload 中神奇的一幕!  

  简单说一下什么是文件上传?所谓文件上传就是将我们本地的资源文件通过网络传输发送到服务器上。

  文件上传的原理其实是IO流实现的,通过二进制流的方式向服务器传输数据,服务器再通过流读取到数据,然后解析成文件,最终保存到服务器上。

  commons-fileupload 工具包主要是我们用来操作文件上传的小助手,里面封装了对流操作的全过程,大大简化了我们实现文件上传的代码复杂度,只需合理的运用类中的方法就可以达到文件上传的效果。

  下面我们通过一个简单的图片上传小例子带大家走进 commons-fileupload 世界,领略这个小助手给我们带来的神奇效果。
  
  点击这里获取 commons-fileupload源码和jar包。

  图片上传小实例三步走:

    一、热身环节

    二、精彩展示

    三、扩展延伸





一、热身环节

  在讲这个图片上传的小实例之前,得先准备两样东西:两个 jar 包、一个页面。

1、两个 jar 包

  commons-fileupload.jar:这是一个核心 jar包,实现文件上传的核心类及方法都在这里面。

  commons-io.jar:这是一个依赖 jar包,是为了辅助 commons-fileupload 包,此包主要是进行 IO操作的,是 commons-fileupload 的依赖包。

2、一个页面

  准备一个 jsp页面,此页面中要有一个 form表单,此表单的特征如下:

表单的 method 属性值必须是 post;
表单的 enctype 属性值必须是 multipart/form-data;
表单中必须提供至少一个 <input type="file"> 这样的文件上传项。


二、精彩展示

  我们需要建立一个 Servlet 用来处理 表单中的数据,在表单中将 type属性值为file 的数据在以下代码中称为“文件数据项”,其他的称为“普通数据项”。在此 Servlet 中需要涉及到三个类,分别是 DiskFileItemFactory类ServletFileUpload类FileItem类。(小伙伴们在导包操作时,注意要选择导入 commons-fileupload.jar 哦~

  DiskFileitemFactory类负责管理磁盘文件,ServletFileUpload类负责上传和解析文件,FileItem类负责保存每个表单数据项的信息。

1、Servlet 具体代码的实现:

@Override
protected void service(HttpServletRequest request, HttpServletResponse response)
        throws ServletException, IOException {
    // 判断是普通表单,还是文件上传表单
    if (!ServletFileUpload.isMultipartContent(request)) {
        throw new RuntimeException("不是文件上传表单!");
    }

    // 创建上传所需要的两个对象
    DiskFileItemFactory factory = new DiskFileItemFactory();  // 磁盘文件对象
    ServletFileUpload sfu = new ServletFileUpload(factory);   // 文件上传对象

    // 设置解析文件上传中的文件名的编码格式
    sfu.setHeaderEncoding("utf-8");

    // 创建 list容器用来保存 表单中的所有数据信息
    List<FileItem> items = new ArrayList<FileItem>();

    // 将表单中的所有数据信息放入 list容器中
    try {
        items = sfu.parseRequest(request);
    } catch (FileUploadException e) {
        e.printStackTrace();
    }

    // 遍历 list容器,处理 每个数据项 中的信息
    for (FileItem item : items) {
        // 判断是否是普通项
        if (item.isFormField()) {
            // 处理 普通数据项 信息
            handleFormField(item);
        } else {
            // 处理 文件数据项 信息
            handleFileField(item);
        }
    }
}

2、两个处理“数据项”信息的方法

  一个用来处理“普通数据项”信息的,另一个是用来处理“文件数据项”信息的。

  1)处理“普通数据项”信息的方法
/**
 * 处理 普通数据项
 * @param item
 */
private void handleFormField(FileItem item) {
    // 获取 普通数据项中的 name值
    String fieldName = item.getFieldName();

    // 获取 普通数据项中的 value值
    String value = "";
    try {
        value = item.getString("utf-8");  // 以 utf-8的编码格式来解析 value值
    } catch (UnsupportedEncodingException e) {
        e.printStackTrace();
    }

    // 输出到控制台
    System.out.println("fieldName:" + fieldName + "--value:" + value);
}
  2)处理“文件数据项”信息的方法
/**
 * 处理 文件数据项
 * @param item
 */
private void handleFileField(FileItem item) {
    // 获取 文件数据项中的 文件名
    String fileName = item.getName();

    // 判断 此文件的文件名是否合法
    if (fileName==null || "".equals(fileName)) {
        return;
    }

    // 控制只能上传图片
    if (!item.getContentType().startsWith("image")) {
        return;
    }

    // 将文件信息 输出到控制台
    System.out.println("fileName:" + fileName);         // 文件名
    System.out.println("fileSize:" + item.getSize());   // 文件大小

    // 获取 当前项目下的 /files 目录的绝对位置
    String path = this.getServletContext().getRealPath("/files");
    File file = new File(path);   // 创建 file对象

    // 创建 /files 目录
    if (!file.exists()) {
        file.mkdir();
    }

    // 将文件保存到服务器上(UUID是通用唯一标识码,不用担心会有重复的名字出现)
    try {
        item.write(new File(file.toString(), UUID.randomUUID()+"_"+fileName));
    } catch (Exception e) {
        e.printStackTrace();
    }
}


三、扩展延伸

1、文件上传的常见限制要求

  1)限制上传文件类别。如:只能上传图片
// 控制只能上传图片
if (!item.getContentType().startsWith("image")) {
    return;
}
  2)限制上传文件大小。如:单个文件大小不能超过 10KB,总文件大小不能超过 100KB
// 创建上传所需要的两个对象
DiskFileItemFactory factory = new DiskFileItemFactory();  // 磁盘文件对象
ServletFileUpload sfu = new ServletFileUpload(factory);   // 文件上传对象

// 限制单个文件的大小
sfu.setFileSizeMax(1024*10);
// 限制上传的总文件大小
sfu.setSizeMax(1024*100);

// 创建 list容器用来保存 表单中的所有数据信息
List<FileItem> items = new ArrayList<FileItem>();

// 将表单中的所有数据信息放入 list容器中
try {
    items = sfu.parseRequest(req);
} catch (FileUploadException e) {
    System.out.println("单个文件大小不能超过10KB,总文件大小不能超过100KB");
    e.printStackTrace();
}

2、common-fileupload 包中三大核心类

  1)DiskFileItemFactory类

作用:磁盘文件工厂类,它还可以设置缓冲区大小以及设置临时保存位置。

setSizeThreshold( int sizeThreshold ) 
    // 设置缓冲区大小,默认sizeThreshold的大小为:10240(10KB)。 
setRepository( File repository ); 
    // 设置临时文件的保存位置,如果不设置,repository为系统的临时目录。 
DiskFileItemFactory(); 
    // 构造方法,缓冲区大小为默认sizeThreshold和临时文件为目录为默认repository的磁盘文件工厂类。 
DiskFileItemFactory( int SizeThreshold , File repository ); 
    // 构造方法,指定缓冲区大小和指定临时文件的磁盘文件工厂类。
  2)ServletFileUpload类

作用:文件上传类,实现上传的一些实用方法的集合。

public List<FileItem> parseRequest( HttpSevletRequest request ); 
    // 解析request对象,获取表单中的所有数据信息,并返回一个List<FileItem>集合,
    // 其中每个FileItem就是一个表单数据项信息。
boolean isMultipartContext( HttpServletRequest request ); 
    // 根据请求对象来判断是普通表单,还是文件上传表单,如果是文件上传表单,则返回true,否则false。
setFileSizeMax( long fileSizeMax ); 
    // 设置单个文件的上传的大小上限。 
setSizeMax( long sizeMax ); 
    // 设置总文件上传的大小上限。 
setHeaderEncoding( Charset charset); 
    // 解决上传文件中的文件名中文乱码的问题。 
ServletFileUpload( DiskFileItemFactory factory ); 
    // 构造方法,指定磁盘文件工厂类,并使用factory指定的缓冲区大小和临时文件。 
  3)FileItem类

作用:表单数据项类,保存了表单数据项的所有信息,以下是完整的 FileItem类对象信息。

普通数据项: name=null, StoreLocation=XXX.tmp, size=5 bytes, isFormField=true, FieldName=username

文件数据项: name=要要.jpg, StoreLocation=XXX.tmp, size=8739 bytes, isFormField=false, FieldName=file2

isFormField(); 
    // 判断是否为普通数据项。 
String getFieldName(); 
    // 获取该表单数据项的名称。即:<input>标签中的name属性。 
String getString( String encoding ); 
    // 以指定的编码格式解析该表单数据项的值。即:<input>标签中的value属性。 
String getName();
    // 获取上传文件中的文件名。 
getInputStream(); 
    // 获取上传文件的内容的输入流,使用文件复制就能完成文件的上传。 
delete();
    // 删除临时文件。

3、解决中文乱码问题

  1)上传的文件名乱码

  使用 ServletFileUpload类 中的 setHeaderEncoding( String encoding ) 方法。

  2)普通表单数据项的内容乱码

  使用 FileItem 中的 getString( String encoding ) 方法。




                        <li class="tool-item tool-active is-like "><a href="javascript:;"><svg class="icon" aria-hidden="true">
                            <use xlink:href="#csdnc-thumbsup"></use>
                        </svg><span class="name">点赞</span>
                        <span class="count">6</span>
                        </a></li>
                        <li class="tool-item tool-active is-collection "><a href="javascript:;" data-report-click="{&quot;mod&quot;:&quot;popu_824&quot;}"><svg class="icon" aria-hidden="true">
                            <use xlink:href="#icon-csdnc-Collection-G"></use>
                        </svg><span class="name">收藏</span></a></li>
                        <li class="tool-item tool-active is-share"><a href="javascript:;"><svg class="icon" aria-hidden="true">
                            <use xlink:href="#icon-csdnc-fenxiang"></use>
                        </svg>分享</a></li>
                        <!--打赏开始-->
                                                <!--打赏结束-->
                                                <li class="tool-item tool-more">
                            <a>
                            <svg t="1575545411852" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="5717" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200"><defs><style type="text/css"></style></defs><path d="M179.176 499.222m-113.245 0a113.245 113.245 0 1 0 226.49 0 113.245 113.245 0 1 0-226.49 0Z" p-id="5718"></path><path d="M509.684 499.222m-113.245 0a113.245 113.245 0 1 0 226.49 0 113.245 113.245 0 1 0-226.49 0Z" p-id="5719"></path><path d="M846.175 499.222m-113.245 0a113.245 113.245 0 1 0 226.49 0 113.245 113.245 0 1 0-226.49 0Z" p-id="5720"></path></svg>
                            </a>
                            <ul class="more-box">
                                <li class="item"><a class="article-report">文章举报</a></li>
                            </ul>
                        </li>
                                            </ul>
                </div>
                            </div>
            <div class="person-messagebox">
                <div class="left-message"><a href="https://blog.csdn.net/sun8112133">
                    <img src="https://profile.csdnimg.cn/3/F/5/3_sun8112133" class="avatar_pic" username="sun8112133">
                                            <img src="https://g.csdnimg.cn/static/user-reg-year/2x/2.png" class="user-years">
                                    </a></div>
                <div class="middle-message">
                                        <div class="title"><span class="tit"><a href="https://blog.csdn.net/sun8112133" data-report-click="{&quot;mod&quot;:&quot;popu_379&quot;}" target="_blank">小异常</a></span>
                                            </div>
                    <div class="text"><span>发布了158 篇原创文章</span> · <span>获赞 164</span> · <span>访问量 15万+</span></div>
                </div>
                                <div class="right-message">
                                            <a href="https://im.csdn.net/im/main.html?userName=sun8112133" target="_blank" class="btn btn-sm btn-red-hollow bt-button personal-letter">私信
                        </a>
                                                            <a class="btn btn-sm  bt-button personal-watch" data-report-click="{&quot;mod&quot;:&quot;popu_379&quot;}">关注</a>
                                    </div>
                            </div>
                    </div>
    

    版权声明:本文为 小异常 原创文章,非商用自由转载-保持署名-注明出处,谢谢!
    本文网址:https://blog.csdn.net/sun8112133/article/details/79968573

    易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
    该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!