rxjava+retrofit实现多图上传实例代码

作者:咚咚淌淌 时间:2022-06-16 18:55:33 

在看了网上多篇rxjava和retrofit的文章后,大概有了一个初步的认识,刚好要做一个多图上传的功能,就拿它开刀吧。下面的内容将基于之前实现方式和使用rxjava实现之间的异同展开,初次写笔记不喜就喷。

普通版多图上传

由于目前手机照片动辄几M的大小,如果不做处理就直接上传,我就笑笑不说话(给个眼神你自己体会)。所以,上传分为两步:对图片进行压缩和请求上传。下面请看伪代码(PS:自己不会写后台,项目后台不能拿来用,所以只能给伪代码了)


//图片集合
List<String> imgs = new ArrayList<>();
//压缩后的图片路径集合
List<String> tmpImgs = new ArrayList<>();

Handler mHandler = new Handler() {
 @Override
 public void handleMessage(Message msg) {
   super.handleMessage(msg);
   //TODO 收到消息后调用网络请求上传
 }
};

public void compressImages() {
 new Thread(new Runnable() {
   @Override
   public void run() {
     for (String path : imgs) {
       //TODO 调用压缩图片的方法,压缩后保存在一个临时文件夹中
       tmpImgs.add("压缩后路径");
     }
     mHandler.sendEmptyMessage(0);
   }
 }).start();
}

看完后是不是觉得很麻烦,好吧可能仅仅是我实现的麻烦而已。都说使用rxjava后逻辑链会变得更清晰,就看看是不是这样,下面请看用rxjava后的伪代码:


@Multipart
@POST("your address")
Observable<String> uploadImgs(@PartMap Map<String, RequestBody> map,                     @Part("imgs") MultipartBody body);

//先定义一个请求接口,除了图片可能还有其他一些参数需要上传,所以还定义了个map。接下来开始正文:

public void upload() {
 final Map<String, RequestBody> map = new HashMap<>();
 map.put("userId", RequestBody.create(MediaType.parse("form-data"),"1");
 final MultipartBody.Builder builder = new MultipartBody.Builder();
 Observable.from(imgs)
     .map(new Func1<String, String>() {
       @Override
       public String call(String path) {
         //调用图片压缩,返回压缩后路径tmp_path
         //注意,Filedata是后台给你的对应的字段
         builder.addFormDataPart("Filedata", "avatar.png", RequestBody.create(MultipartBody.FORM, new File(tmp_path)));
         return path;
       }
     }).last()
     .flatMap(new Func1<String, Observable<String>>() {
       @Override
       public Observable<String> call(String path) {
         return apiService.uploadImgs(map, builder.build());
       }
     })
     .subscribeOn(Schedulers.io())
     .observeOn(AndroidSchedulers.mainThread())
     .subscribe(new Subscriber<String>() {
       @Override
       public void onCompleted() {
       }
       @Override
       public void onError(Throwable e) {
         //错误处理
       }
       @Override
       public void onNext(String res) {
         //成功后处理
       }
     });
}

黑人问号脸?代码看起来还是很多啊,你TM在逗我。听本汪开始胡说八道:

1、首先定义个Map,这个就是用来上传其他参数用的,为什么value是RequestBody类型的,用String不就可以了吗,瞎装什么逼啊。好吧,本汪开始也是这么认为的,结果传到服务器的值自带‘'加成,传个1过去变成了‘1',正打算一本正经的找后台谈谈的,发现自己传上去的就是这样(脸红ing)。然后发现用@part注解的,如果不使用RequestBody,会自动加上‘',这点至今不知为何,还请懂的小伙伴释疑。

2、然后是MultipartBody.Builder,顾名思义,能添加多个RequestBody,用来添加多个图片。好了,小火车要开动了。

3、简单说下接下来这一大段代码是干嘛的,当然建立在你已经了解rxjava的from、map、flatmap、last是用来干嘛的基础上。

a、from会将imgs集合拆分成单个的String发送出去

b、map的作用是在此进行图片压缩,并将压缩后的图片添加到MultipartBody.Builder,相当于for循环压缩了图片。

c、flatmap这里,可谓是成败再次一举了。这里有一个转换,注意map处理后返回的String依然是一个String类型,经过flatmap后将转化为 Observable<String>,也就是我们图片上传后返回的结果。

d、好了,到此为止好像已经达到我们一条链下来就实现了图片上传的功能了,感觉是要清晰那么一点(如果没有,那我还TM瞎折腾什么)。哎,别走啊你把last忽略掉是什么鬼。

e、如果不在map后添加last方法,大家可以试一试,保证后台白眼都要翻到天上去了。由于from一个一个的发送,所以每一个对象都会在flatmap这里调用一次uploadImgs方法,这样肯定是不行了,加last方法后,只会发送发送从map出来的序列的最后一个对象,这样就保证在所有图片都压缩完成并且加入后MultipartBody.Builder后再调用uploadImgs方法,并且只会调用一次。

来源:http://www.jianshu.com/p/6b0cbcda7e9c

标签:rxjava,retrofit,上传
0
投稿

猜你喜欢

  • Android studio实现简单的计算器

    2022-09-07 23:23:28
  • 详解JAVAEE——SSH三大框架整合(spring+struts2+hibernate)

    2022-09-20 04:41:04
  • C++容器适配与栈的实现及dequeque和优先级详解

    2023-11-02 12:57:52
  • Android 广播大全 Intent Action 事件详解

    2021-09-10 12:40:15
  • SpringCloud中分析讲解Feign组件添加请求头有哪些坑梳理

    2023-05-01 19:30:11
  • App内切换语言详解

    2023-04-12 11:21:44
  • C#实现12306自动登录的方法

    2023-11-07 13:20:27
  • 使用C#调用百度地图并实现坐标点的设置以及读取示例

    2023-01-06 02:32:57
  • Java将对象保存到文件中/从文件中读取对象的方法

    2022-06-18 21:26:42
  • Android 基于agora 开发视频会议的代码

    2021-11-30 02:53:04
  • Android编程之SQLite数据库操作方法详解

    2021-11-27 03:42:21
  • java中addMouseListener()方法的使用

    2021-07-07 19:29:35
  • C++ vector的简单实现

    2023-04-09 17:13:02
  • Iconfont(矢量图标)+iconmoon(图标svg互转)配合javascript实现社交分享系统

    2023-09-28 13:23:19
  • SpringBoot 如何使用RestTemplate发送Post请求

    2022-03-03 09:35:47
  • Android 使用压缩纹理的方案

    2023-09-26 12:25:23
  • SpringBoot Knife4j在线API文档框架基本使用

    2022-03-10 21:27:48
  • Kotlin 使用Lambda来设置回调的操作

    2021-07-22 03:08:04
  • Java操作文件输出为字符串以及字符串输出为文件的方法

    2022-02-01 19:32:45
  • Android Studio 多层级 Module 对 aar 引用问题解决方法

    2023-08-06 19:41:27
  • asp之家 软件编程 m.aspxhome.com