前面一篇文章写了实现照相功能的一个例子,其实那个实现效果是个略缩图。要查看全图就要先指定照片的存放路径。以后我会修改那个文章。今天先说下图片,文件等上传的实现。接着拿照片说事,光照完了不行还得往服务器上传。
我们做web开发的时候几乎都是通过一个表单来实现上传。并且是post的方式。而且都必须要加个参数enctype = "multipart/form-data".然后再上传后台用各种框架里的插件之类的就可以接收了,并没有关心过这个文件具体是怎么传的。现在用android开发 没有那些框架了,所以不得不关心一下了。
其实我们这种前后台的交互是用的HTTP协议。而http协议默认是传的字符串。所以我们上传文件的话要加enctype = "multipart/form-data"这个参数来说明我们这传的是文件不是字符串了。而我们做web开发的时候,浏览器是自动解析HTTP协议的。里面传的哪些东西我们不用管。只要记住几个参数就行。而我们要上传的文件报文是保存在请求的头文件里面的。下面就是上传文件头文件的格式:
POST/logsys/home/uploadIspeedLog!doDefault.html HTTP/1.1
Accept: text/plain, */*
Accept-Language: zh-cn
Host: 192.168.24.56
Content-Type:multipart/form-data;boundary=-----------------------------7db372eb000e2
User-Agent: WinHttpClient
Content-Length: 3693
Connection: Keep-Alive
-------------------------------7db372eb000e2
Content-Disposition: form-data; name="file"; filename="kn.jpg"
Content-Type: image/jpeg
(此处省略jpeg文件二进制数据...)
-------------------------------7db372eb000e2--
这就是Http上传发送的文件格式。而我们要发送的时候必然要遵循这种格式来并且不能出一点差错包括每行后面的回车,下面一段文字是网上找的感觉写的比较精彩。(尊重原创:原文地址)
红色字体部分就是协议的头。给服务器上传数据时,并非协议头每个字段都得说明,其中,content-type是必须的,它包括一个类似标志性质的名为boundary的标志,它可以是随便输入的字符串。对后面的具体内容也是必须的。它用来分辨一段内容的开始。Content-Length: 3693 ,这里的3693是要上传文件的总长度。绿色字体部分就是需要上传的数据,可以是文本,也可以是图片等。数据内容前面需要有Content-Disposition, Content-Type以及Content-Transfer-Encoding等说明字段。最后的紫色部分就是协议的结尾了。
注意这一行:
Content-Type: multipart/form-data; boundary=---------------------------7db372eb000e2
根据 rfc1867, multipart/form-data是必须的.
---------------------------7db372eb000e2 是分隔符,分隔多个文件、表单项。其中b372eb000e2 是即时生成的一个数字,用以确保整个分隔符不会在文件或表单项的内容中出现。Form每个部分用分隔符分割,分隔符之前必须加上"--"着两个字符(即--{boundary})才能被http协议认为是Form的分隔符,表示结束的话用在正确的分隔符后面添加"--"表示结束。
前面的 ---------------------------7d 是 IE 特有的标志,Mozila 为---------------------------71.
每个分隔的数据的都可以用Content-Type来表示下面数据的类型,可以参考rfc1341 (http://www.ietf.org/rfc/rfc1341.txt)
以上对上传文件的头文件格式讲的应该比较清楚了。其实很多参数我们也用不到,也不必刻意去记住,下面就看一个例子。应该能更好的理解:
class="dp-j" start="1">
- package com.example.photo;
-
- import java.io.BufferedReader;
- import java.io.DataOutputStream;
- import java.io.File;
- import java.io.FileInputStream;
- import java.io.IOException;
- import java.io.InputStream;
- import java.io.InputStreamReader;
- import java.io.OutputStream;
- import java.net.HttpURLConnection;
- import java.net.MalformedURLException;
- import java.net.URL;
- import java.util.UUID;
-
-
- public class HttpAssist {
- private static final String TAG = "uploadFile";
- private static final int TIME_OUT = 10 * 10000000;
- private static final String CHARSET = "utf-8";
- public static final String SUCCESS = "1";
- public static final String FAILURE = "0";
-
- public static String uploadFile(File file) {
- String BOUNDARY = UUID.randomUUID().toString();
- String PREFIX = "--", LINE_END = "\r\n";
- String CONTENT_TYPE = "multipart/form-data";
- String RequestURL = "http://192.168.0.100:7080/YkyPhoneService/Uploadfile1";
- try {
- URL url = new URL(RequestURL);
- HttpURLConnection conn = (HttpURLConnection) url.openConnection();
- conn.setReadTimeout(TIME_OUT);
- conn.setConnectTimeout(TIME_OUT);
- conn.setDoInput(true);
- conn.setDoOutput(true);
- conn.setUseCaches(false);
- conn.setRequestMethod("POST");
- conn.setRequestProperty("Charset", CHARSET);
- conn.setRequestProperty("connection", "keep-alive");
- conn.setRequestProperty("Content-Type", CONTENT_TYPE + ";boundary="
- + BOUNDARY);
- if (file != null) {
-
-
-
- OutputStream outputSteam = conn.getOutputStream();
-
- DataOutputStream dos = new DataOutputStream(outputSteam);
- StringBuffer sb = new StringBuffer();
- sb.append(PREFIX);
- sb.append(BOUNDARY);
- sb.append(LINE_END);
-
-
-
-
-
- sb.append("Content-Disposition: form-data; name=\"img\"; filename=\""
- + file.getName() + "\"" + LINE_END);
- sb.append("Content-Type: application/octet-stream; charset="
- + CHARSET + LINE_END);
- sb.append(LINE_END);
- dos.write(sb.toString().getBytes());
- InputStream is = new FileInputStream(file);
- byte[] bytes = new byte[1024];
- int len = 0;
- while ((len = is.read(bytes)) != -1) {
- dos.write(bytes, 0, len);
- }
- is.close();
- dos.write(LINE_END.getBytes());
- byte[] end_data = (PREFIX + BOUNDARY + PREFIX + LINE_END)
- .getBytes();
- dos.write(end_data);
- dos.flush();
-
-
-
- int res = conn.getResponseCode();
- if (res == 200) {
- return SUCCESS;
- }
- }
- } catch (MalformedURLException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- }
- return FAILURE;
- }
- }
这是文件上传的客户端代码。 认真读一下代码,应该能够理解上传文件的格式了。同样写好这个头文件之后采用HttpURLConnection向后台发送。也就是浏览器所用的http协议。我们只是把这个协议自己手动调用并且手动填写头文件内容。而不是通过浏览器帮我们写了。看这段代码的时候我发现都是把字符串转化成字节流,然后利用DataOutputStream这个类来想后台传输。图片文件也是利用这个类向后台传。不知道大家有没有跟我想法一样的觉得到后台之后通过Request获取字节流,然后把流写到相应格式的文件或者图片里就收到了。我觉得理论上应该是这样的。而且网上也有一些相应的文件或者代码。但是我拿来用的话却都不成功,以下是一个例子(不能用。)
- public class UploadFile extends HttpServlet {
-
- static int i = 0;
- public void doPost(HttpServletRequest request, HttpServletResponse response)
- throws IOException {
- ServletInputStream in = request.getInputStream();
- byte[] buffer = new byte[1024];
- File file = new File(request.getSession().getServletContext().getRealPath("/img/"),"img_"+i+".jpg");
- i++;
- FileOutputStream out = new FileOutputStream(file);
- int len = in.read(buffer, 0, 1024);
- while( len!=-1){
- out.write(buffer,0,len);
- len = in.read(buffer, 0, 1024);
- }
- out.close();
- in.close();
- }
- }
我觉得这几行代码跟我的想法很相似,但就是运行不成功。向后台上传一个图片之后也收到了,但就是打不开,提示文件损坏。而且我觉得这种思路是对的,于是就一直在找类似的实现方法或者,按自己的思路修改一下代码,带最终还是不行。所以只好用其他的方法。
服务器接收上传的文件的方法最后是通过利用apache提供的两个jar包来实现的。commons-fileupload.jar和commons-io.jar这俩jar包。在服务器端添加这俩包之后,写一个Servlet来实现文件接收。直接上代码:
- public class Uploadfile1 extends HttpServlet {
-
- @Override
- public void doPost(HttpServletRequest request, HttpServletResponse response)
- throws ServletException, IOException {
- request.setCharacterEncoding("utf-8");
-
- DiskFileItemFactory factory = new DiskFileItemFactory();
-
- String path = request.getSession().getServletContext().getRealPath("/upload");
-
- factory.setRepository(new File(path));
-
- factory.setSizeThreshold(1024*1024);
-
- ServletFileUpload upload = new ServletFileUpload(factory);
-
- try{
-
- List<FileItem> list = (List<FileItem>)upload.parseRequest(request);
- for(FileItem item:list){
-
- String name = item.getFieldName();
-
- if(item.isFormField()){
-
- String value = item.getString();
- request.setAttribute(name, value);
- }
-
- else{
-
- String value = item.getName();
-
- int start = value.lastIndexOf("\\");
-
- String filename = value.substring(start+1);
- request.setAttribute(name, filename);
-
-
-
-
- OutputStream out = new FileOutputStream(new File(path,filename));
- InputStream in = item.getInputStream();
-
- int length = 0;
- byte[] buf = new byte[1024];
- System.out.println("获取文件总量的容量:"+ item.getSize());
-
- while((length = in.read(buf))!=-1){
- out.write(buf,0,length);
- }
- in.close();
- out.close();
- }
- }
- }catch(Exception e){
- e.printStackTrace();
- }
-
- }
- }
代码同样转自网络,具体地址给忘了,这个真不是故意的。。。
代码上有相应注释,应该都能看懂。并且这个开源jar包提供的方法挺强大的,支持多文件上传之类的。我猜它的源码同样是获取客户端传过来的字节流。后面的代码跟上面提供的思路一样。只是不知道它的Item究竟是如何获得这个字节流的。按着源码看了看 也没看太明白。以后再慢慢研究吧。有哪位明白的 还请指点。
这样利用HTTP做到文件的上传和接收都已经正确运行了。先留下个笔记,以后用的时候可以看看。