前面一篇文章写了实现照相功能的一个例子,其实那个实现效果是个略缩图。要查看全图就要先指定照片的存放路径。以后我会修改那个文章。今天先说下图片,文件等上传的实现。接着拿照片说事,光照完了不行还得往服务器上传。

我们做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)

以上对上传文件的头文件格式讲的应该比较清楚了。其实很多参数我们也用不到,也不必刻意去记住,下面就看一个例子。应该能更好的理解:

  1. package com.example.photo;
  2. import java.io.BufferedReader;
  3. import java.io.DataOutputStream;
  4. import java.io.File;
  5. import java.io.FileInputStream;
  6. import java.io.IOException;
  7. import java.io.InputStream;
  8. import java.io.InputStreamReader;
  9. import java.io.OutputStream;
  10. import java.net.HttpURLConnection;
  11. import java.net.MalformedURLException;
  12. import java.net.URL;
  13. import java.util.UUID;
  14. public class HttpAssist {
  15. private static final String TAG = "uploadFile";
  16. private static final int TIME_OUT = 10 * 10000000; // 超时时间
  17. private static final String CHARSET = "utf-8"; // 设置编码
  18. public static final String SUCCESS = "1";
  19. public static final String FAILURE = "0";
  20. public static String uploadFile(File file) {
  21. String BOUNDARY = UUID.randomUUID().toString(); // 边界标识 随机生成
  22. String PREFIX = "--", LINE_END = "\r\n";
  23. String CONTENT_TYPE = "multipart/form-data"; // 内容类型
  24. String RequestURL = "http://192.168.0.100:7080/YkyPhoneService/Uploadfile1";
  25. try {
  26. URL url = new URL(RequestURL);
  27. HttpURLConnection conn = (HttpURLConnection) url.openConnection();
  28. conn.setReadTimeout(TIME_OUT);
  29. conn.setConnectTimeout(TIME_OUT);
  30. conn.setDoInput(true); // 允许输入流
  31. conn.setDoOutput(true); // 允许输出流
  32. conn.setUseCaches(false); // 不允许使用缓存
  33. conn.setRequestMethod("POST"); // 请求方式
  34. conn.setRequestProperty("Charset", CHARSET); // 设置编码
  35. conn.setRequestProperty("connection", "keep-alive");
  36. conn.setRequestProperty("Content-Type", CONTENT_TYPE + ";boundary="
  37. + BOUNDARY);
  38. if (file != null) {
  39. /**
  40. * 当文件不为空,把文件包装并且上传
  41. */
  42. OutputStream outputSteam = conn.getOutputStream();
  43. DataOutputStream dos = new DataOutputStream(outputSteam);
  44. StringBuffer sb = new StringBuffer();
  45. sb.append(PREFIX);
  46. sb.append(BOUNDARY);
  47. sb.append(LINE_END);
  48. /**
  49. * 这里重点注意: name里面的值为服务器端需要key 只有这个key 才可以得到对应的文件
  50. * filename是文件的名字,包含后缀名的 比如:abc.png
  51. */
  52. sb.append("Content-Disposition: form-data; name=\"img\"; filename=\""
  53. + file.getName() + "\"" + LINE_END);
  54. sb.append("Content-Type: application/octet-stream; charset="
  55. + CHARSET + LINE_END);
  56. sb.append(LINE_END);
  57. dos.write(sb.toString().getBytes());
  58. InputStream is = new FileInputStream(file);
  59. byte[] bytes = new byte[1024];
  60. int len = 0;
  61. while ((len = is.read(bytes)) != -1) {
  62. dos.write(bytes, 0, len);
  63. }
  64. is.close();
  65. dos.write(LINE_END.getBytes());
  66. byte[] end_data = (PREFIX + BOUNDARY + PREFIX + LINE_END)
  67. .getBytes();
  68. dos.write(end_data);
  69. dos.flush();
  70. /**
  71. * 获取响应码 200=成功 当响应成功,获取响应的流
  72. */
  73. int res = conn.getResponseCode();
  74. if (res == 200) {
  75. return SUCCESS;
  76. }
  77. }
  78. } catch (MalformedURLException e) {
  79. e.printStackTrace();
  80. } catch (IOException e) {
  81. e.printStackTrace();
  82. }
  83. return FAILURE;
  84. }
  85. }

这是文件上传的客户端代码。 认真读一下代码,应该能够理解上传文件的格式了。同样写好这个头文件之后采用HttpURLConnection向后台发送。也就是浏览器所用的http协议。我们只是把这个协议自己手动调用并且手动填写头文件内容。而不是通过浏览器帮我们写了。看这段代码的时候我发现都是把字符串转化成字节流,然后利用DataOutputStream这个类来想后台传输。图片文件也是利用这个类向后台传。不知道大家有没有跟我想法一样的觉得到后台之后通过Request获取字节流,然后把流写到相应格式的文件或者图片里就收到了。我觉得理论上应该是这样的。而且网上也有一些相应的文件或者代码。但是我拿来用的话却都不成功,以下是一个例子(不能用。)

  1. public class UploadFile extends HttpServlet {
  2. static int i = 0;
  3. public void doPost(HttpServletRequest request, HttpServletResponse response)
  4. throws IOException {
  5. ServletInputStream in = request.getInputStream();
  6. byte[] buffer = new byte[1024];
  7. File file = new File(request.getSession().getServletContext().getRealPath("/img/"),"img_"+i+".jpg");
  8. i++;
  9. FileOutputStream out = new FileOutputStream(file);
  10. int len = in.read(buffer, 0, 1024);
  11. while( len!=-1){
  12. out.write(buffer,0,len);
  13. len = in.read(buffer, 0, 1024);
  14. }
  15. out.close();
  16. in.close();
  17. }
  18. }

我觉得这几行代码跟我的想法很相似,但就是运行不成功。向后台上传一个图片之后也收到了,但就是打不开,提示文件损坏。而且我觉得这种思路是对的,于是就一直在找类似的实现方法或者,按自己的思路修改一下代码,带最终还是不行。所以只好用其他的方法。

服务器接收上传的文件的方法最后是通过利用apache提供的两个jar包来实现的。commons-fileupload.jar和commons-io.jar这俩jar包。在服务器端添加这俩包之后,写一个Servlet来实现文件接收。直接上代码:

  1. public class Uploadfile1 extends HttpServlet {
  2. @Override
  3. public void doPost(HttpServletRequest request, HttpServletResponse response)
  4. throws ServletException, IOException {
  5. request.setCharacterEncoding("utf-8");
  6. //获得磁盘文件条目工厂。
  7. DiskFileItemFactory factory = new DiskFileItemFactory();
  8. //获取文件上传需要保存的路径,upload文件夹需存在。
  9. String path = request.getSession().getServletContext().getRealPath("/upload");
  10. //设置暂时存放文件的存储室,这个存储室可以和最终存储文件的文件夹不同。因为当文件很大的话会占用过多内存所以设置存储室。
  11. factory.setRepository(new File(path));
  12. //设置缓存的大小,当上传文件的容量超过缓存时,就放到暂时存储室。
  13. factory.setSizeThreshold(1024*1024);
  14. //上传处理工具类(高水平API上传处理?)
  15. ServletFileUpload upload = new ServletFileUpload(factory);
  16. try{
  17. //调用 parseRequest(request)方法  获得上传文件 FileItem 的集合list 可实现多文件上传。
  18. List<FileItem> list = (List<FileItem>)upload.parseRequest(request);
  19. for(FileItem item:list){
  20. //获取表单属性名字。
  21. String name = item.getFieldName();
  22. //如果获取的表单信息是普通的文本信息。即通过页面表单形式传递来的字符串。
  23. if(item.isFormField()){
  24. //获取用户具体输入的字符串,
  25. String value = item.getString();
  26. request.setAttribute(name, value);
  27. }
  28. //如果传入的是非简单字符串,而是图片,音频,视频等二进制文件。
  29. else{
  30. //获取路径名
  31. String value = item.getName();
  32. //取到最后一个反斜杠。
  33. int start = value.lastIndexOf("\\");
  34. //截取上传文件的 字符串名字。+1是去掉反斜杠。
  35. String filename = value.substring(start+1);
  36. request.setAttribute(name, filename);
  37. /*第三方提供的方法直接写到文件中。
  38. * item.write(new File(path,filename));*/
  39. //收到写到接收的文件中。
  40. OutputStream out = new FileOutputStream(new File(path,filename));
  41. InputStream in = item.getInputStream();
  42. int length = 0;
  43. byte[] buf = new byte[1024];
  44. System.out.println("获取文件总量的容量:"+ item.getSize());
  45. while((length = in.read(buf))!=-1){
  46. out.write(buf,0,length);
  47. }
  48. in.close();
  49. out.close();
  50. }
  51. }
  52. }catch(Exception e){
  53. e.printStackTrace();
  54. }
  55. }
  56. }

代码同样转自网络,具体地址给忘了,这个真不是故意的。。。

代码上有相应注释,应该都能看懂。并且这个开源jar包提供的方法挺强大的,支持多文件上传之类的。我猜它的源码同样是获取客户端传过来的字节流。后面的代码跟上面提供的思路一样。只是不知道它的Item究竟是如何获得这个字节流的。按着源码看了看 也没看太明白。以后再慢慢研究吧。有哪位明白的 还请指点。

这样利用HTTP做到文件的上传和接收都已经正确运行了。先留下个笔记,以后用的时候可以看看。

Android 上传文件,图片。以及服务器端接收相关。的更多相关文章

  1. android上传文件到服务器

    package com.spring.sky.image.upload.network; import java.io.DataOutputStream; import java.io.File; i ...

  2. android -上传文件到服务器

    android上传文件到服务器       重点:最好是设置好content-type这些参数的配置!     package com.spring.sky.image.upload.network; ...

  3. android 上传文件

    android对于上传文件,还是非常easy的,和java里面的上传都是一样的,基本上都是熟悉操作输出流和输入流!另一个特别重要的就是须要一些content-type这些參数的配置!  假设这些都弄好 ...

  4. Android上传文件至服务器(上)

    每一次都不能上首页,真悲催..管理员让我上一次首页? 很多时候我更愿意一个人写代码,与其在垃圾代码上改改改,我更愿意直接重构. 整洁的代码简单直接.整洁的代码如同优美的散文.整洁的代码从不隐藏设计者的 ...

  5. 【经验记录】Android上传文件到服务器

    Android中实现上传文件,其实是很简单的,和在java里面是一样的,基本上都是熟悉操作输出流和输入流!还有一个特别重要的就是需要配置content-type的一些参数!如果这些都弄好了,上传就很简 ...

  6. Android上传文件到服务器(转)

    Android中实现上传文件,其实是很简单的,和在java里面是一样的,基本上都是熟悉操作输出流和输入流!还有一个特别重要的就是需要配置content-type的一些参数!如果这些都弄好了,上传就很简 ...

  7. ASP.NET、JAVA跨服务器远程上传文件(图片)的相关解决方案整合

    一.图片提交例: A端--提交图片 protected void Page_Load(object sender, EventArgs e) { if (!IsPostBack) { string u ...

  8. PHP socket上传文件图片

    最近了解了下下socket方面的东西,想做一个socket上传文件的例子. 在网上搜了搜代码执行后,图片数据传输了一半,图片的下半部分是灰色的.然后就自己仿着搜来的代码和php.net 中socket ...

  9. Android上传文件之FTP

    android客户端实现FTP文件(包括图片)上传应该没什么难度.写下来就了为了记录一下,望能帮到新手. 需要用到 commons-net-3.0.1.jar,后面附上jar包. 直接上代码: /** ...

随机推荐

  1. centos7没有ifconfig命令

    yum search ifconfig 看到 net-tools.x86_64 : Basic networking tools 安装net-tools: yum install net-tools

  2. 使用instantclient_11_2 和PL/SQL Developer工具包连接oracle 11g远程数据库(转)

    1,先到Oracle网站下载Instant Client : http://www.oracle.com/technology/global/cn/software/tech/oci/instantc ...

  3. Server Performance Advisor (SPA) 3.0

    http://blogs.technet.com/b/windows-server-china-blog/archive/2013/03/26/server-performance-advisor-s ...

  4. websocket与canvas[转]

    server端还是用tomcat7的方式客户端 <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"  ...

  5. angularjs显示html片段

    ngBindHtml <div ng-controller="ExampleController"> <p ng-bind-html="myHTML&q ...

  6. Java多线程之syncrhoized内置互斥锁的用法详解

       转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/5827547.html    解决并行冲突最有效的方法就是加同步锁,主要有以下几种方法:   1:动态方法 ...

  7. tensorflow serving 编写配置文件platform_config_file的方法

    1.安装grpc gRPC 的安装: $ pip install grpcio 安装 ProtoBuf 相关的 python 依赖库: $ pip install protobuf 安装 python ...

  8. java HMAC_SHA1加密算法

      java HMAC_SHA1加密算法 CreationTime--2018年7月14日16点46分 Author:Marydon 1.准备工作 import javax.crypto.Mac; i ...

  9. Spring Cloud Edgware Release Notes

    Spring Cloud Edgware builds on Spring Boot 1.5.x. Renamed starters A number of starters did not foll ...

  10. 解决linux的centos版本修改时间重启后无效的问题

    安装完centos后,发现时间与本地时间不匹配,在网上找了好多的办法,但是一直没有奏效,重启之后,又恢复为原来的时间.很是纳闷.最后抱着试一试的心态加上了这句指令: ln -sf /usr/share ...