文件的上传和下载

来源:互联网 发布:主奴社交软件 编辑:程序博客网 时间:2024/04/28 17:02

1、文件上传的原理分析

1.1文件上传的必要前提:

     a、提供form表单,method必须是post

     bform表单的enctype必须是multipart/form-data

     c、提供input type="file"类的上传输入域


/**  *例:JSP文件中  */<form enctype = “multipart/form-data” action =“${pageContext.request.contextPath}/servlet/uploadSerlet1”  method=”post”>   <input type “text” name = “name”/><br/>   <input type = “file” name = “photo”/> <br/>   <input type = “submit” value = “上传”/><br/></form>


/** *uploadServlet1.java 文件 */public class UploadServlet1 extends HttpServlet{     public void doGet(HttpServletRequest request,HttpServletResponse response)throws ServletException,IOexception{/*由于表单中提交数据的方式改为multipart/form-data,所以request.getParamter(name);失效String name = request.getParameter(“name”);String photo = request.getParameter(“photo”);System.out.println(name);System.out.println(photo);}*/InputStream is = request.getInputStream();Int len = 0;byte[] b = new byte[1024];While((len = is.read(b))!= -1){System.out.println(new String);}     Public void doPost(HttpServletRequest request , HttpServletResponse response)throws ServletException , IOexception{}}


 服务器获取数据:request.getParameter(String)方法获取指定的表单字段字符内容,但文件上传表单已经不在是字符内容,而是字节内容,所以失效。


2、借助第三方的上传组件实现文件上传

2.1 fileupload概述
    fileupload是由apachecommons组件提供的上传组件。它最主要的工作就是帮我们解析request.getInputStream()
        导入commons-fileupload相关jar

l commons-fileupload.jar,核心包;

l commons-io.jar,依赖包。

2.2 fileupload的核心类有:
    DiskFileItemFactoryServletFileUpload(核心类)FileItem
      解析原理
DiskFileItemFactory可以加工ServletFileupload.ServletFileupload可以操作FileItem(表单项)
表单里上传的用户名,上传的文件,都叫FileItem
                          不同的FileItem有不同的方法
2.2 fileupload简单应用

使用fileupload组件的步骤如下:

1. 创建工厂类DiskFileItemFactory对象:

DiskFileItemFactory factory = new DiskFileItemFactory()

2. 使用工厂创建解析器对象:

ServletFileUpload fileUpload = new ServletFileUpload(factory)

3. 使用解析器来解析request对象:

List<FileItem> list = fileUpload.parseRequest(request)


FileItem对象对应一个表单项(表单字段)。可以是文件字段或普通字段

       boolean isFormField():判断当前表单字段是否为普通文本字段,如果返回false,说明是文件字段;

      String getFieldName():获取字段名称,例如:<input type=”text” name=”username”/>,返回的是username

      String getString():获取字段的内容,如果是文件字段,那么获取的是文件内容,当然上传的文件必须是文本文件;

      String getName():获取文件字段的文件名称;(a.txt)

      String getContentType():获取上传的文件的MIME类型,例如:text/plain。 

       int getSize():获取上传文件的大小;

       InputStream getInputStream():获取上传文件对应的输入流;

       void write(File):把上传的文件保存到指定文件中。

      delete();


代码案例

package com.itheima.upload;import java.io.File;import java.io.FileOutputStream;import java.io.IOException;import java.io.InputStream;import java.io.UnsupportedEncodingException;import java.util.List;import java.util.UUID;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.FileUploadBase;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.FilenameUtils;public class UploadServlet2 extends HttpServlet {public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {// request.setCharacterEncoding("UTF-8");// 要执行文件上传的操作// 判断表单是否支持文件上传。即:enctype="multipart/form-data"boolean isMultipartContent = ServletFileUpload.isMultipartContent(request);if (!isMultipartContent) {throw new RuntimeException("your form is not multipart/form-data");}// 创建一个DiskFileItemfactory工厂类DiskFileItemFactory factory = new DiskFileItemFactory();factory.setRepository(new File("f:\\"));  // 指定临时文件的存储目录// 创建一个ServletFileUpload核心对象ServletFileUpload sfu = new ServletFileUpload(factory);// 解决上传表单项乱码问题sfu.setHeaderEncoding("UTF-8");// 限制上传文件的大小// 解析request对象,并得到一个表单项的集合try {// sfu.setFileSizeMax(1024*1024*3);//表示3M大小// sfu.setSizeMax(1024*1024*6);List<FileItem> fileItems = sfu.parseRequest(request);// 遍历表单项数据for (FileItem fileitem : fileItems) {if (fileitem.isFormField()) {// 是普通表单项processFormField(fileitem);} else {// 是上传表单项processUploadField(fileitem);}}} catch (FileUploadBase.FileSizeLimitExceededException e) {// throw new RuntimeException("文件过在,不能超过3M");System.out.println("文件过在,不能超过3M");} catch (FileUploadBase.SizeLimitExceededException e) {System.out.println("总文件大小不能超过6M");} catch (FileUploadException e) {e.printStackTrace();}}                //是上传表单项private void processUploadField(FileItem fileitem) {try {// 得到文件输入流InputStream is = fileitem.getInputStream();// 创建一个文件存盘的目录String directoryRealPath = this.getServletContext().getRealPath("/WEB-INF/upload");File storeDirectory = new File(directoryRealPath);// 即代表文件又代表目录if (!storeDirectory.exists()) {storeDirectory.mkdirs();// 如果不存在,创建一个指定的目录}// 得到上传的名子String filename = fileitem.getName();// 文件项中的值 F:\图片素材\小清新\43.jpg 或者// 43.jpgif (filename != null) {// filename =// filename.substring(filename.lastIndexOf(File.separator)+1);filename = FilenameUtils.getName(filename);// 效果同上}// 解决文件同名的问题filename = UUID.randomUUID() + "_" + filename;// 目录打散// String childDirectory = makeChildDirectory(storeDirectory); //// 2015-10-19String childDirectory = makeChildDirectory(storeDirectory, filename); // a/b// 上传文件,自动删除临时文件fileitem.write(new File(storeDirectory, childDirectory+ File.separator + filename));fileitem.delete();} catch (IOException e) {e.printStackTrace();} catch (Exception e) {e.printStackTrace();}}// 是上传表单项private void processUploadField1(FileItem fileitem) {try {// 得到文件输入流InputStream is = fileitem.getInputStream();// 创建一个文件存盘的目录String directoryRealPath = this.getServletContext().getRealPath("/WEB-INF/upload");File storeDirectory = new File(directoryRealPath);// 即代表文件又代表目录if (!storeDirectory.exists()) {storeDirectory.mkdirs();// 创建一个指定的目录}// 得到上传的名子String filename = fileitem.getName();// 文件项中的值 F:\图片素材\小清新\43.jpg 或者// 43.jpg// 处理文件名/* * 解决此问题: F:\\apache-tomcat-7.0.52\\webapps\\day18_00_upload\\upload\\F:\\图片素材\\小清新\\41.jpg */if (filename != null) {// filename =// filename.substring(filename.lastIndexOf(File.separator)+1);filename = FilenameUtils.getName(filename);// 效果同上}// 解决文件同名的问题filename = UUID.randomUUID() + "_" + filename;// 目录打散// String childDirectory = makeChildDirectory(storeDirectory); // 2015-10-19String childDirectory = makeChildDirectory(storeDirectory, filename); // 2015-10-19// 在storeDirectory下,创建完整目录下的文件File file = new File(storeDirectory, childDirectory+ File.separator + filename); // 绝对目录/日期目录/文件名// 通过文件输出流将上传的文件保存到磁盘FileOutputStream fos = new FileOutputStream(file);int len = 0;byte[] b = new byte[1024];while ((len = is.read(b)) != -1) {fos.write(b, 0, len);}fos.close();is.close();fileitem.delete();} catch (IOException e) {e.printStackTrace();}}// 目录打散private String makeChildDirectory(File storeDirectory, String filename) {int hashcode = filename.hashCode();// 返回字符转换的32位hashcode码System.out.println(hashcode);String code = Integer.toHexString(hashcode); // 把hashcode转换为16进制的字符// abdsaf2131safsdSystem.out.println(code);String childDirectory = code.charAt(0) + File.separator+ code.charAt(1); // a/b// 创建指定目录File file = new File(storeDirectory, childDirectory);if (!file.exists()) {file.mkdirs();}return childDirectory;}// 按日期打散/* * private String makeChildDirectory(File storeDirectory) { *  * SimpleDateFormat sdf =new SimpleDateFormat("yyyy-MM-dd"); String * dateDirectory = sdf.format(new Date()); //只管创建目录 File file = new * File(storeDirectory,dateDirectory); if(!file.exists()){ file.mkdirs(); } *  * return dateDirectory; } */// 普通表单项private void processFormField(FileItem fileitem) {try {String fieldname = fileitem.getFieldName();// 字段名String fieldvalue = fileitem.getString("UTF-8");// 字段值//fieldvalue = new String(fieldvalue.getBytes("iso-8859-1"),"utf-8");System.out.println(fieldname + "=" + fieldvalue);} catch (UnsupportedEncodingException e) {e.printStackTrace();}}public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {doGet(request, response);}}


3、文件上传时要考虑的几个问题(经验分享)

         a、保证服务器的安全

         把保存上传文件的目录放在用户直接访问不到的地方。

        

 

         b、避免文件被覆盖

         让文件名唯一即可

        


         c、避免同一个文件夹中的文件过多

        方案一:按照日期进行打散存储目录

         

  方案二:用文件名的hashCode计算打散的存储目录:二级目录

      

 

       d、限制文件的大小:web方式不适合上传大的文件

       单个文件大小:

       ServletFileUpload.setFileSizeMax(字节)

      总文件大小:(多文件上传)

      ServletFileUpload.setSizeMax(字节)

 

      e、上传字段用户没有上传的问题

     通过判断文件名是否为空即可


      f、临时文件的问题

      DiskFileItemFactory:

  作用:产生FileItem对象 

  内部有一个缓存,缓存大小默认是10Kb。如果上传的文件超过10Kb,用磁盘作为缓存。

      存放缓存文件的目录在哪里?默认是系统的临时目录。

 

  如果自己用IO流实现的文件上传,要在流关闭后,清理临时文件。

      FileItem.delete();



       

0 0