您的位置:首页 > 理论基础 > 计算机网络

android 使用OkHttp上传多张图片

2016-04-11 21:14 836 查看

   简述

还是先来说说为啥用OkHttp作为多图片上传的框架,原因有两点:

1、OkHttp可以作为Volley底层传输协议,速度更快

2、使用Xutils和KJFramework上传图片存在一个小问题,首先,可以上传,并且可以上传多张图片,也可以上传其他的参数,那问题在哪里呢?在后台接受参数时很不灵活,Xutlis及KJFramework使用HashMap来上传每个参数,每一张图片也必须有一个唯一的key,上传一张图片就要定义一个参数来接收,上传两张图片就要定义两个参数来接收,当上传的图片数量不确定的时候,如最多9张或者16张,后台接受图片的时候就要定义9个或者16个,这样的方式很不利于扩展,最好是一个参数接收所有所有图片,不会因为这种不确定的问题,就去定义很多的参数,然后一个个判断是否存在。OkHttp底层则不是这样,大概的浏览了下源码,底层接收参数的时候使用的是List,只要使用相同的key就可以添加到同一个list,而后台只需要根据这一个key不断遍历就行,无论多少张图片都无障碍,也没有了后顾之忧。

核心代码实现

<code class="hljs cs has-numbering"><span class="hljs-comment">//参数类型</span>
<span class="hljs-keyword">private</span> <span class="hljs-keyword">static</span> final MediaType MEDIA_TYPE_PNG = MediaType.parse(<span class="hljs-string">"image/png"</span>);
<span class="hljs-comment">//创建OkHttpClient实例</span>
<span class="hljs-keyword">private</span> final OkHttpClient client = <span class="hljs-keyword">new</span> OkHttpClient();

MultipartBuilder builder = <span class="hljs-keyword">new</span> MultipartBuilder().type(MultipartBuilder.FORM);

<span class="hljs-comment">//遍历map中所有参数到builder</span>
<span class="hljs-keyword">for</span> (String key : map.keySet()) {
builder.addFormDataPart(key, map.<span class="hljs-keyword">get</span>(key));
}

<span class="hljs-comment">//遍历paths中所有图片绝对路径到builder,并约定key如“upload”作为后台接受多张图片的key</span>
<span class="hljs-keyword">for</span> (String path : paths) {
builder.addFormDataPart(<span class="hljs-string">"files"</span>, path, RequestBody.create(MEDIA_TYPE_PNG, <span class="hljs-keyword">new</span> File(path)));
}

<span class="hljs-comment">//构建请求体</span>
RequestBody requestBody = builder.build();

<span class="hljs-comment">//构建请求</span>
Request request = <span class="hljs-keyword">new</span> Request.Builder()
.url(url)<span class="hljs-comment">//地址</span>
.post(requestBody)<span class="hljs-comment">//添加请求体</span>
.build();

<span class="hljs-comment">//发送异步请求,同步会报错,Android4.0以后禁止在主线程中进行耗时操作</span>
client.newCall(request).enqueue(<span class="hljs-keyword">new</span> Callback() {
@Override
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">onFailure</span>(Request request, IOException e) {
System.<span class="hljs-keyword">out</span>.println(<span class="hljs-string">"request = "</span> + request.urlString());
System.<span class="hljs-keyword">out</span>.println(<span class="hljs-string">"e.getLocalizedMessage() = "</span> + e.getLocalizedMessage());
}

@Override
<span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">onResponse</span>(Response response) throws IOException {
<span class="hljs-comment">//看清楚是response.body().string()不是response.body().toString()</span>
System.<span class="hljs-keyword">out</span>.println(<span class="hljs-string">"response = "</span> + response.body().<span class="hljs-keyword">string</span>());
}
});</code>

基本实现思路就是这样了,更多请参考OkHttp文档,项目中我准备完全除去Xutils的其他部分,如文件下载部分。

知识拓展

List与HashMap区别

List可以保存多个相同或者不同的元素

HashMap则是以键值对(key-value)保存元素,当添加多个相同key的元素,之前的元素会被覆盖

这一点至关重要,Xutils以及KJFframework的设计都没有考虑到这一点,看似很好用,但当我要添加多张相同或者不同的图片文件作为参数传输给后台,我却需要定义很多的key,不然无论添加多少张图片结果却只能是最后一张

 

 

SpingMvc 上传图片核心代码:

 @RequestMapping(value="uploadPic")

 public @ResponseBody Messege uploadPic(@RequestParam(required = false) MultipartFile[] files,HttpServletRequest request){ 

  Messege messege=new Messege();

  String path = request.getSession().getServletContext()

    .getRealPath("/upload")

    + File.separator;

  System.out.println(path);

  if (files != null && files.length > 0) {

   for (MultipartFile f : files) {

    if (f.isEmpty()) {

     continue;

    } else {

     String fileName = f.getOriginalFilename();// 真实文件 名

     System.out.println(f.getContentType());

     try {

      FileUtils.copyInputStreamToFile(f.getInputStream(),

        new File(path + fileName));

      

      messege.setRet_code(200);

      messege.setRet_msg("上传图片成功");

     } catch (IOException e) {

      messege.setRet_code(0);

      messege.setRet_msg("上传图片");

      e.printStackTrace();

     }

    }

   }

  }

 

  

  

  

  return messege;

  

  

  

 }

spring-servlet.xml 配置文件

<bean id="multipartResolver"

  class="org.springframework.web.multipart.commons.CommonsMultipartResolver">

  <property name="maxUploadSize" value="10485760000" />

 </bean>

 

 

 
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: