现在是做的一个上传图片的功能,上传图片用的一个flash插件,flash插件模拟得是form表单的提交方式。我在后台servlet通过request.getInputStream()得到的流 ,如果上传图片选择非中文名称可以将上传的图片打开,但是如果选择中文名称的图片上传,后台处理生成的图片打不开。请教高手该怎么解决。
代码参照的这篇博客:http://blog.sina.com.cn/s/blog_48a6d7ba0100uft7.html
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); // 根据长度,将消息实体的内容读入字节数组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); // 再读一行信息,是这一部分数据的Content-type s = reqbuf.readLine(); // 设置文件输入流,准备写文件 f = new File("d:" + 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; }
} } 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)); } }
代码参照的这篇博客:http://blog.sina.com.cn/s/blog_48a6d7ba0100uft7.html
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); // 根据长度,将消息实体的内容读入字节数组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); // 再读一行信息,是这一部分数据的Content-type s = reqbuf.readLine(); // 设置文件输入流,准备写文件 f = new File("d:" + 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; }
} } 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)); } }
fileRealName = s; fileRealName = URLDecoder.decode(fileRealName , "UTF-8"); //加这句,try,catch根据情况加下。 out.print("fileRealName=" + fileRealName + "<br>");