上传文件 流解析request.getInputStream()
引用:http://blog.sina.com.cn/s/blog_48a6d7ba0100uft7.html
-------------------------------------------------------------------------------------------------
浏览器 采用了一种编码方式,即 "multipart/form-data" 的编码方式,采用这种方式,浏览器可以很容易将表单内的数据和文件放在一起发送.这 种编码方式先定义好一个不可能在数据中出现的字符串作为 分界符,然后用它将各个数据段分开,而对于每个数据段都对应着 HTML 页面表单 中的一个 Input 区,包括一个 content-disposition 属性,说明了这个数据段的一些信息,如果这个数据段的内容是一个文件,还会有 Content-Type 属性,然后就是数据本身. 我们可以用request.getInputStream()或request.getReader()得到 提交的数据.
但想要得到文件内容,需要我们自己解析................
-------------------------------------------------------------------------------------------------
package com.csv.test;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.io.StringReader;
import java.util.HashMap;
import java.util.Map;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import com.sun.org.apache.xalan.internal.xsltc.runtime.Hashtable;
@SuppressWarnings("serial")
public class CsvTest extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
this.doPost(request, response);
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=UTF-8");
PrintWriter out = response.getWriter();
final int NONE = 0;//状态码,表示没有特殊操作
final int DATAHEADER = 1;//表示下一行要读到报头信息
final int FILEDATA = 2;//表示下面要读的是上传文件和二进制数据
final int FIELDDATA = 3;//表示下面要读到表单域的文本值
//请求消息实体的总长度(请求消息中队消息头之外的数据长度)
int totalbytes = request.getContentLength();
File f;//上传文件储存在服务器上
//容纳请求消息实体的字节数组
byte[] dataOrigin = new byte[totalbytes];
//对于post多个文件的表单,b作为原始数据的副本提取文件数据的操作
byte[] b = new byte[totalbytes];
//请求消息类型
String contentType = request.getContentType();
String fieldname = "";//表单的名称
String fieldvalue = "";//表单的值
String fileFormName = "";//上传的文件再表单中的名称
String fileRealName = "";//上传文件 的真实名字
String boundary = "";//分界字符串
String lastboundary = "";//结束分界字符串
int fileSize = 0;//文件长度
//容纳表单域的名称、值的哈希表
Map<String ,String> formfieldsTable = new HashMap<String,String>();
//容纳文件域的名称、文件名的哈希表
Map<String ,String> filenameTable = new HashMap<String,String>();
//在消息头类型中找到分界符的定义
int pos = contentType.indexOf("boundary=");
int pos2 ;//position2
if(pos != -1){
pos += "boundary=".length();
boundary = "--"+contentType.substring(pos);
lastboundary = boundary +"--";//得到结束分界符
}
int state =NONE;//起始状态为NONE
//得到请求消息的数据入流
DataInputStream in = new DataInputStream(request.getInputStream());
in.readFully(dataOrigin);//根据长度,将消息实体的内容读入字节数组
in.close();//关闭数据流
String reqcontent = new String(dataOrigin); //从字节数组中得到表示实体的字符串
//从字符串中得到输出缓冲流
BufferedReader reqbuf = new BufferedReader(new StringReader(reqcontent));
//设置循环标志
boolean flag = true;
//int i = 0;
while(flag == true){
String s = reqbuf.readLine();
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=");
//先判断出这是一个文本表单域的头信息,还是一个上传文件 的头信息
if(pos == -1){
//如果是文本表单域的头信息,解析出表单域的名称
pos = s.indexOf("name=");
pos += "name=".length()+1;//1表示后面的”的占位
s = s.substring(pos);
int l = s.length();
s =s.substring(0,l-1);//应该是"
fieldname = s;//表单域的名称放入fieldname
out.print("fieldname="+fieldname);
state = FIELDDATA;//设置状态码,准备读取表单域的值
}else{
//如果是文件数据的头,先存储这一行,用于在字节数组中定位
String temp = s;
//先解析出文件名
pos = s.indexOf("name=");
pos += "name=".length()+1;//1表示后面的"的占位
pos2 = s.indexOf("filename=");
String s1 = s.substring(pos,pos2-3);//3表示";加上一个空格
fileFormName = s1;
pos2 += "filename=".length()+1;//1表示后面的“的占位
s = s.substring(pos2);
int l = s.length();
s = s.substring(0,l-1);
pos2 = s.lastIndexOf("\\");//对于IE浏览器的设置
s = s.substring(pos2+1);
fileRealName = s;
out.print("fileRealName="+fileRealName+"<br>");
out.print("fileRealName.length()="+fileRealName.length()+"<br>");
if(fileRealName.length() != 0){//确定文件被上传
//下面这一部分从字节数组中取出文件的数据
b = dataOrigin;//复制原始数据以便提取文件
pos = byteIndexOf(b,temp,0);//定位行
//定位下一行,2表示一个回车和一个换行占两个字节
b = subBytes(b,pos+temp.getBytes().length+2,b.length);
//再读一行信息,是这一部分数据的Context-type
s = reqbuf.readLine();
//设置文件输入流,准备写文件
f = new File("d:/img"+File.separator +fileRealName);
DataOutputStream 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);//将文件数据存盘
fileout.close();
fileSize = b.length -1;//文件长度丰入fileSize
out.print("fileFormName="+fileFormName +"filename="+fileRealName+"fileSize="+fileSize+"<br>");
filenameTable.put(fileFormName, fileRealName);
state = FILEDATA;
}
}
break;
case FIELDDATA:
//读取表单域的值
s = reqbuf.readLine();
fieldvalue = s;//存入fieldvalue
out.print("fieldvalue="+fieldvalue+"<br>");
formfieldsTable.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;
}
}
//指定内容类型,并且可以显示中文
out.print("<HTML>");
out.print("<HEAD><TITLE>文件上传结果</TITLE></HEAD>");
out.print("<BODY>");
out.print("<H1>文件上传结果</H1><hr>");
out.print("ID为"+formfieldsTable.get("FileID1")+"的文件"+filenameTable.get("FileData1")+"已经上传!<br>");
out.print("ID为"+formfieldsTable.get("FileID2")+"的文件"+filenameTable.get("FileData2")+"已经上传!<br>");
out.print("</BODY>");
out.print("</HTML>");
// response.setContentType("application/x-javascript;charset=UTF-8");
// PrintWriter out = response.getWriter();
//
// out.println("恭喜,上传成功。");
// out.close();
}
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;
}
//在b中找到s的第一个元素
search:for(i = start; i<= max ; i++){
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;
}
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));
}
}