当前位置: 移动技术网 > IT编程>移动开发>Android > rxjava+retrofit实现多图上传实例代码

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

2019年07月24日  | 移动技术网IT编程  | 我要评论

猜字谜七十二小时,善德女王国语,九重紫书书网

在看了网上多篇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方法,并且只会调用一次。

以上就是我用rxjava+retrofit做多图上传的小笔记,希望对大家的学习有所帮助,也希望大家多多支持移动技术网。

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网