Servlet实现文件上传

package utils;

import java.io.BufferedReader;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.PrintWriter;
import java.io.StringReader;
import java.util.Hashtable;

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

public class UploadFileServlet extends HttpServlet {
 private static final long serialVersionUID = 1L;
      
  
    public UploadFileServlet() {
        super();
    }

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

 
 public void doPost(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {
  

           //注意:在数据部分的分界符比在Content-Type中规定的分界符在前面多了两个短杠"--",而在所有数据结束时分界符前后都多了两个短杠"--"
     final int NONE = 0; // 状态码,表示没有特殊操作
     final int DATAHEADER = 1; // 表示下一行要读到报头信息
     final int FILEDATA = 2; // 表示下面要读到表单域的文本值
     final int FILEDDATA = 3; // 表示下面要读的是上传文件和二进制数据

     // 请求消息实体的总长度(请求消息中除消息头之外的数据长度)
     int totalbytes = req.getContentLength();
     // 容纳请求消息实体的字节数组
     byte[] b = new byte[totalbytes];
     // 请求消息类型
     String contentType = req.getContentType();
     String fieldname = ""; // 表单域的名称
     String fieldvalue = ""; // 表单域的值
     String filename = ""; // 上传的文件名称
     String boundary = ""; // 分界符字符串
     String lastboundary = ""; // 结束分界符字符串
     int fileSize = 0; // 文件长度
     // 容纳表单域的名称/值的哈希表
     Hashtable formfields = new Hashtable();

     // 在消息头类型中找到分界符的定义
     System.out.println("contentType-------------->------>"+contentType);
     int pos = contentType.indexOf("boundary=");
     String fileID; // 上传的文件ID
     if (pos != -1) {
      pos += "boundary=".length();
      boundary = "--" + contentType.substring(pos);// 解析出分界符
      lastboundary = boundary + "--";// 得到结束分界符
     }
    
     //System.out.println("contentType.substring(pos)-------------->------>"+contentType.substring(pos).length());
    // System.out.println("boundary-------------->------>"+boundary.length());
    // System.out.println("lastboundary-------------->------>"+lastboundary.length());
     int state = NONE; // 起始状态为NONE
     // 得到请求消息的数据输入流
     DataInputStream in = new DataInputStream(req.getInputStream());
     DataOutputStream fileout=null;
     in.readFully(b); // 根据长度,将消息实体的内容读入字节数组b中
     in.close(); // 关闭数据流
     String reqcontent = new String(b); // 从字节数组中得到表示实体的字符串
     //String reqcontent = new String(b,"gbk");
     // 从字符串中得到输出缓冲流
     BufferedReader reqbuf = new BufferedReader(new StringReader(reqcontent));
     // 设置循环标志
     boolean flag = true;
     int i = 0;
     //int thl=0;
     while (flag == true) {
      String s = reqbuf.readLine();
     // System.out.println("s"+(thl++)+"-------------->------>"+s);
      if (s == lastboundary || s == null)
       break;
      switch (state) {
      case NONE:
       if (s.startsWith(boundary)) {
        // 如果读到分界符,则表示下一行一个头信息
        state = DATAHEADER;
        i += 1;
       }
   break;
      case DATAHEADER:
       pos = s.indexOf("filename=");
      
       //System.out.println("s-------------->------>"+s);
      
       int l = 0;
       // 先判断出这是一个文本表单域的头信息,还是一个上传文件的头信息
       if (pos == -1) {
        // 如果是文本表单域的头信息,解析出表单域的名称
        pos = s.indexOf("name=");
        pos += "name=".length() + 1;// 1表示后面的"的占位
        s = s.substring(pos);
        l = s.length();
        s = s.substring(0, l - 1);
        fieldname = s;// 表单域的名称放入fieldname   
        state = FILEDDATA; // 设置状态码,准备读取表单域的值
       } else {
        // 如果是文件数据的头,先存储这一行,用于在字节数组中定位
        String temp = s;   
        // 先解析出文件名
        pos = s.indexOf("filename=");
        pos += "filename=".length() + 1;// 1表示后面的"的占位
        s = s.substring(pos);
        l = s.length();
        s = s.substring(0, l - 1);
        pos = s.lastIndexOf("\\");
        s = s.substring(pos + 1);    
        filename = s;// 文件名存入filename   
        // filename=new String(filename.getBytes(),"utf-8");
        //System.out.println("flename-------------->------>"+filename);
        // 下面这一部分从字节数组中取出文件的数据
        pos = byteIndexOf(b, temp, 0); // 定位行
        // 定位下一行,2表示一个回车和一个换行占2个字节
        b = subBytes(b, pos + temp.getBytes().length + 2, b.length);
        // 再读一行信息,是这一部分数据的Content-type
        s = reqbuf.readLine();
        // 设置文件输入流,准备写文件
        String path=getServletContext().getRealPath("/");
    
        //File f = new File("f://",filename);
        File f = new File(path+"xls//",filename);
//        if(f.exists()){
//         f.delete();
//        }
        f.createNewFile();
         fileout = new DataOutputStream(
          new FileOutputStream(f));
        // 字节数组再往下一行,4表示两个回车换行占4个字节,本行的回车
        // 换行2个字节,Content-type的下一行是回车换行表示的空行,占2个字节
        // 得到文件数据的起始位置
        b = subBytes(b, s.getBytes().length + 4, b.length);
        pos = byteIndexOf(b, boundary, 0);// 定位文件数据的结尾
        b = subBytes(b, 0, pos - 1);// 取得文件数据

        fileout.write(b, 0, b.length - 1);// 将文件数据存盘
        fileSize = b.length - 1; // 文件长度存入fileSize
        state = FILEDATA;
     
       }
      
       break;
      case FILEDDATA:
       // 读出表单域的值
       s = reqbuf.readLine();
       fieldvalue = s;// 存入fieldvalue 
       formfields.put(fieldname, fieldvalue);
       state = NONE;
       break;
      case FILEDATA:
       // 如果是文件数据不进行分析,直接读过去
       while ((!s.startsWith(boundary))
         && (!s.startsWith(lastboundary))) {
        s = reqbuf.readLine(); 
        if (s.startsWith(boundary)) {
         state = DATAHEADER;
        } else
         break;
        break;
       }

      fileout.close();
      
      }
     }
     // 指定内容类型,并且可显示中文
     res.setContentType("text/html;charset=gb2312");
     PrintWriter out = res.getWriter();
     out.println("<HTML>");
     out.println("<HEAD><TITLE>文件上传结果</TITLE></HEAD>");
     out.println("<BODY>");
     out.println("<H1>文件上传结果</H1><hr>");
     out.println("ID为" + formfields.get("FileID") + "的文件" + filename
       + "已经上传!" + "文件长度为:" + fileSize + "字节");
     out.println("</BODY>");
     out.println("</HTML>");
    
  }

  /**
  * 字节数组中的s串起始索引
  */
  private static int byteIndexOf(byte[] b, String s, int start) {
     return byteIndexOf(b, s.getBytes(), start);
  }

  private static int byteIndexOf(byte[] b, byte[] s, int start) {
     int i;
     if (s.length == 0) {
      return 0;
     }
     int max = b.length - s.length;
     if (max < 0)
      return -1;
     if (start > max)
      return -1;
     if (start < 0)
      start = 0;

     search: for (i = start; i <= max; i++) {
      // 在b中找到s的第一个元素
      if (b[i] == s[0]) {
       // 找到了s中的第一个元素后,比较剩余的部分是否相等
       int k = 1;
       while (k < s.length) {
        if (b[k + i] != s[k]) {
         continue search;
        }
        k++;
       }
       return i;
      }
     }
     return -1;
  }

  /**
  * 从b中提取子串
  */
  private static byte[] subBytes(byte[] b, int from, int end) {
     byte[] result = new byte[end - from];
     System.arraycopy(b, from, result, 0, end - from);
     return result;
  }

  private static String subBytesString(byte[] b, int from, int end) {
     return new String(subBytes(b, from, end));
  }


  
 }


 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值