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


猜你喜欢
- 本文实例讲述了C#转换日期类型的方法。分享给大家供大家参考。具体分析如下:如:将日期1999-5-31 11:20转换成 /Date(928
- 本文实例为大家分享了java通过PDF模板填写PDF表单的具体代码,包括图片,供大家参考,具体内容如下需要用到的java包: it
- 1.概述Spring MVC是Spring Framework的一部分,是基于Java实现MVC的轻量级Web框架。Spring MVC的特
- Java 8 中 Function 接口的介绍Java 8 中提供了一个函数式接口 Function,这个接口表示对一个参数做一些
- @PropertySource作用是:对自定义的properties文件加载使用:@PropertySource(value={"
- 解决My eclipse 工程发布时端口占用问题如果运行后如图的错,需要进行如下操作来解决:a:打开cmd,输入netstat -ano 找
- 本文实例为大家分享了安卓Button按钮的四种点击事件,供大家参考,具体内容如下第一种:内部类实现 1.xml里面先设置Button属性&l
- 在这篇文章中,我们将介绍如下内容:==运算符与基元类型==运算符与引用类型==运算符与String类型==运算符与值类型==运算符与泛型==
- 内发光原理内发光原理简单概况是:采样周边像素alpha取平均值叠加效果。概括来说似乎好像特别简单,但需要一定的理解和消化。发光物体可以当做是
- Kotlin基础教程之Run,标签Label,函数Function-Type在Java中可以使用{}建立一个匿名的代码块,代码块会被正常的执
- 1.依赖maven依赖如下,需要说明的是,spring-boot-starter-data-redis里默认是使用lettuce作为redi
- 使用idea创建javaweb项目idea还是写框架项目比较爽,原生的javaweb项目不是特别方便,这篇文章就是记录一下创建的过程图较多注
- 一、简单介绍翻看Spring的源码时,发现@Bean注解的源码上标注了Since: 3.0,也就是说,@Bean注解是Spring从3.0版
- Java8对List<Integer>的求和想要用流对List<Integer>进行求和,但查找完资料都是对List
- 这个窗口是右下角提示小窗口,主要用于提示。private void btnStartNotification_Click(object se
- 本文实例为大家分享了基于C#实现网页爬虫的详细代码,供大家参考,具体内容如下HTTP请求工具类:功能:1、获取网页html2、下载网络图片u
- 前言本文主要介绍其具体的实现思路(视频仅有代码输入,并无过程介绍等),同时,在原本实现的基础上,进行了多处修改和优化,具体参见下面的内容。优
- 1.for循环import com.google.common.base.Function;import com.google.common
- 1、未配置之前2、开始配置 2.1 新建一个unauth.html<!DOCTYPE html><html la
- cmd调用phantomjs官方资料:http://phantomjs.org/quick-start.html手动执行从官方下载phant