jsp servlet文件上传

一、文件上传的基本操作: 
1、 表单属性enctype的设置 

multipart/form-data和application/x-www-form-urlencoded的区别 

FORM元素的enctype属性指定了表单数据向服务器提交时所采用的编码类型,默认的缺省值是“application/x-www-form-urlencoded”。 

然而,在向服务器发送大量的文本、包含非ASCII字符的文本或二进制数据时这种编码方式效率很低。 

在文件上载时,所使用的编码类型应当是“multipart/form-data”,它既可以发送文本数据,也支持二进制数据上载。 

Browser端<form>表单的ENCTYPE属性值为multipart/form-data,它告诉我们传输的数据要用到多媒体传输协议,由于多媒体传输的都是大量的数据,所以规定上传文件必须是post方法,<input>的type属性必须是file。 
要获取提交过来的数据可以用request.getInputstream()再放到输出流输出发现输出的是

   -----------------------------7dd32c39803b2
   Content-Disposition: form-data; name="username"

   wzhting
   -----------------------------7dd32c39803b2
   Content-Disposition: form-data; name="f1"; filename="C:\Documents and Settings\wzhting\妗岄潰\a.txt"
   Content-Type: text/plain

   aaaaaaaaaaaaaaaaaa
   -----------------------------7dd32c39803b2
   Content-Disposition: form-data; name="f2"; filename="C:\Documents and Settings\wzhting\妗岄潰\b.txt"
   Content-Type: text/plain

   bbbbbbbbbbbbbbbbbbb
   -----------------------------7dd32c39803b2--

这样的话得分割比较麻烦所以apache开发了上传工具
实现过程: 

导入apache的两个jar:commons-fileupload.jar和commons-io.jar
DiskFileItemFactory:设置环境-》ServletFileUpload:核心上传类(主要作用:解析请求的正文内容)-》FileItem:代表表单中的一个输入域。
DiskFileItemFactory:
            public void setSizeThreshold(int?sizeThreshold) :设置缓冲区大小。默认是10Kb。
              当上传的文件超出了缓冲区大小,fileupload组件将使用临时文件缓存上传文件
             public void setRepository(java.io.File repository):设置临时文件的存放目录。默认是系统的临时文件存放目录。
             DiskFileItemFactory(int size,File file);
 
ServletFileUpload:
           boolean isMultipartContent(HttpServletRequest?request):判断用户的表单的enctype是否是multipart/form-data类型的。
           List parseRequest(HttpServletRequest request):解析请求正文中的内容返回List<FileItem>
           setFileSizeMax(4*1024*1024);//设置单个上传文件的大小
          upload.setSizeMax(6*1024*1024);//设置总文件大小
 
FileItem
1.获取普通字段信息:  
 boolean isFormField():是否是普通字段
   String getFieldName:获取普通字段的字段名
   String getString():获取普通字段的值
2.获取上传文件信息:
   InputStream getInputStream():获取上传字段的输入流
   String getName():获取上传的文件名
最普通的文件上传:
 public void doGet(HttpServletRequest request, HttpServletResponse response)
   throws ServletException, IOException {
  response.setContentType("text/html;utf-8");
  response.setCharacterEncoding("utf-8");
  DiskFileItemFactory factory=new DiskFileItemFactory();
  ServletFileUpload load=new ServletFileUpload(factory);
  load.setHeaderEncoding("UTF-8");
  if(!load.isMultipartContent(request)){
   return;
  }
  try {
   List<FileItem> list=load.parseRequest(request);
   for(FileItem item: list){
    if(item.isFormField())
    {
     String name=item.getFieldName();
     String value=item.getString("utf-8");
     response.getWriter().write(name+"="+value);
    }
    else
    {
     String real=getServletContext().getRealPath("/WEB-INF/upload");
     String filename=item.getName();
     filename=filename.substring(filename.lastIndexOf("\\")+1);
     response.getWriter().write("<br />文件名:"+filename);
     FileOutputStream fos=new FileOutputStream(real+"\\"+filename);
     InputStream ins=item.getInputStream();
     byte buf[]=new byte[1024];
     int len=0;
     while((len=ins.read(buf))>0){
      fos.write(buf,0,len);
     }
     ins.close();
     fos.close();
    }
   }
  } catch (FileUploadException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  }
 }
 public void doPost(HttpServletRequest request, HttpServletResponse response)
   throws ServletException, IOException {
  doGet(request, response);

 }

}

 

要注意的9个问题
 1、如何保证服务器的安全
  把保存上传文件的目录放到WEB-INF目录中。
 2、中文乱码问题
  2.1普通字段的中文请求参数
   String value = FileItem.getString("UTF-8");
  2.2上传的文件名是中文
   解决办法:request.setCharacterEncoding("UTF-8");
 3、重名文件被覆盖的问题
   System.currentMillions()+"_"+a.txt(乐观)
   
   UUID+"_"+a.txt:保证文件名唯一
 4、分目录存储上传的文件
  方式一:当前日期建立一个文件夹,当前上传的文件都放到此文件夹中。
  方式二:利用文件名的hash码打散目录来存储。
   int hashCode = fileName.hashCode();
   
    1001 1010 1101 0010 1101 1100 1101 1010
hashCode&0xf;   0000 0000 0000 0000 0000 0000 0000 1111 &
    ---------------------------------------------
    0000 0000 0000 0000 0000 0000 0000 1010   取hashCode的后4位
             0000~1111:整数0~15共16个
    
    1001 1010 1101 0010 1101 1100 1101 1010
(hashCode&0xf0) 0000 0000 0000 0000 0000 0000 1111 0000  &
    --------------------------------------------
    0000 0000 0000 0000 0000 0000 1101 0000  >>4
    --------------------------------------------
    0000 0000 0000 0000 0000 0000 0000 1101
             0000~1111:整数0~15共16个
 5、限制用户上传的文件类型
  通过判断文件的扩展名来限制是不可取的。
  通过判断其Mime类型才靠谱。FileItem.getContentType();
 6、如何限制用户上传文件的大小
  6.1单个文件大小限制。超出了大小友好提示
   抓异常进行提示:org.apache.commons.fileupload.FileUploadBase.FileSizeLimitExceededException
  6.2总文件大小限制。超出了大小友好提示
   抓异常进行提示:org.apache.commons.fileupload.FileUploadBase.SizeLimitExceededException
 7、临时文件的问题
  commons-fileupload组件不会删除超出缓存的临时文件。
  
  FileItem.delete()方法删除临时文件。但一定要在关闭流之后。
 8、多个文件上传时,没有上传内容的问题
  if(fileName==null||"".equals(fileName.trim())){
   continue;
  }
 9、上传进度检测
  给ServletFileUpload注册一个进度监听器即可,把上传进度传递给页面去显示
  //pBytesRead:当前以读取到的字节数
  //pContentLength:文件的长度
  //pItems:第几项
  public void update(long pBytesRead, long pContentLength,
    int pItems) {
   System.out.println("已读取:"+pBytesRead+",文件大小:"+pContentLength+",第几项:"+pItems);
  }

 

完整文件上传代码:

 

public void doPost(HttpServletRequest request, HttpServletResponse response)
   throws ServletException, IOException {
  request.setCharacterEncoding("UTF-8");// 中文文件名乱码问题
  response.setContentType("text/html;charset=UTF-8");
  PrintWriter pout = response.getWriter();
  DiskFileItemFactory df = new DiskFileItemFactory(1024 * 1024, new File(
    "d:" + File.separator + "cache"));
  ServletFileUpload sfu = new ServletFileUpload(df);
  if (!sfu.isMultipartContent(request)) {
   System.out.println("提交类型不对");
  } else {
   try {
    //进度条
/*    sfu.setProgressListener(new ProgressListener() {
     //pBytesRead读取的字节数 pContentLength总长度 pItems读取了第几个
     public void update(long pBytesRead, long pContentLength,
       int pItems) {
      // TODO Auto-generated method stub
      System.out.println("已读取:"+pBytesRead+",文件大小:"+pContentLength+",第几项:"+pItems);
     }
    });*/
    List<FileItem> list = sfu.parseRequest(request);
    // sfu.setFileSizeMax(4 * 1024 * 1024);// 设置单个上传文件的大小
    // sfu.setSizeMax(6 * 1024 * 1024);// 设置总文件大小
    for (FileItem item : list) {
     if (item.isFormField()) {
      System.out.println(item.getFieldName() + ":"
        + item.getString("utf-8"));// form表单参数乱码问题
     } else {
      String mimeType = item.getContentType();
      // if (mimeType.startsWith("image"))
      if (1 == 1) {// 限制上传文件类型只能为图片
       String fileName = item.getName();
       if (fileName == null || "".equals(fileName.trim())) {// 多个文件上传时,没有上传内容的问题
        continue;
       }
       fileName = fileName.substring(fileName
         .lastIndexOf("\\") + 1);
       String storePath = getServletContext().getRealPath(
         "/WEB-INF/Files");
       InputStream is = item.getInputStream();
       fileName = UUID.randomUUID() + "_" + fileName;// 防止文件名重复
       storePath = makeStorePath(storePath, fileName);// 多文件夹存储
       OutputStream fos = new FileOutputStream(storePath
         + File.separator + fileName);
       BufferedOutputStream out = new BufferedOutputStream(
         fos);
       byte b[] = new byte[1024];
       int len = -1;
       while ((len = is.read(b)) != -1) {
        out.write(b, 0, len);
       }
       is.close();
       out.close();
       item.delete();//删除临时文件
      }
     }
    }
   } /*catch (org.apache.commons.fileupload.FileUploadBase.FileSizeLimitExceededException e) {
    // 单个文件超出大小时的异常
    pout.write("单个文件大小不能超出4M");
   } catch (org.apache.commons.fileupload.FileUploadBase.SizeLimitExceededException e) {
    // 总文件超出大小时的异常
    pout.write("总文件大小不能超出6M");
   } */catch (FileUploadException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
   }
   response.sendRedirect("/ServletFileUpload/ok.jsp");
  }
 }

 private String makeStorePath(String storePath, String fileName) {
  // TODO Auto-generated method stub
  int hashCode = fileName.hashCode();
  int dir1 = hashCode & 0xf;// 0000~1111:整数0~15共16个
  int dir2 = (hashCode & 0xf0) >> 4;// 0000~1111:整数0~15共16个
  String newstore = storePath + File.separator + dir1 + File.separator
    + dir2;
  File file = new File(newstore);
  if (!file.exists()) {
   file.mkdirs();// 递归创建目录
  }
  return newstore;
 }


 

 

 

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值