上传文件 流解析request.getInputStream()

简介: 引用:http://blog.sina.com.cn/s/blog_48a6d7ba0100uft7.html ---------------------------------------------------------------------------------------------...

引用: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));
}

 

}

 

 

相关文章
|
6月前
09JavaWeb基础 - Response类案例(下载文件)
09JavaWeb基础 - Response类案例(下载文件)
26 0
|
语音技术
发送octet-stream格式的请求返回乱码处理
octet-stream格式的请求返回乱码处理
|
4月前
Response设置响应数据功能介绍及重定向
Response设置响应数据功能介绍及重定向
35 0
|
9月前
|
前端开发
如何在前端下载后端返回的文件流时,获取请求头中的文件名称?
如何在前端下载后端返回的文件流时,获取请求头中的文件名称?
664 0
|
JavaScript 前端开发 Java
Java http Post下载Excel文件 xxx.xlsx 失败解决,及传输文件类型ContentType对应关系
查询数据库生成Excel下载功能。本来很简单的功能却搞了挺久的,主要因为下载的文件打不开,且此功能已是基本功能。这里记录下来方便后面查阅。 ContentType对应的文件关系查看下面连接 https://www.runoob.com/http/mime-types.html
211 0
|
Java 开发者
使用 Response 下载文件 | 学习笔记
快速学习使用 Response 下载文件,介绍了使用 Response 下载文件系统机制, 以及在实际应用过程中如何使用。
276 0
使用 Response 下载文件 | 学习笔记
|
Java
通过后台Http请求的形式实现文件的上传与传输
在开发中,我们可能会遇到两个不同的系统之间进行文件传输的需求,由于没有对应的界面操作,所以,此时就需要通过java后台代码的形式来实现文件的上传操作,本文主要讲解如何通过后台方法的形式来实现文件的上传功能,并能实现相应的接收功能。
1110 0
|
Java
Java 技术篇 - ServerSocket接收http的url请求中包含中文的处理方法,URLDecode与URLEncode,url解码与编码
Java 技术篇 - ServerSocket接收http的url请求中包含中文的处理方法,URLDecode与URLEncode,url解码与编码
235 0
Java 技术篇 - ServerSocket接收http的url请求中包含中文的处理方法,URLDecode与URLEncode,url解码与编码