RxAndroid的基础使用

article/2025/9/29 22:26:30

作为一个android开发者,在开发应用的过程中避免不了异步这个问题。android系统为我们提供了Handler这个类帮助我们进行线程间的通信和切换,但是GitHub上也有很多其他非常优秀的开源框架来帮助我们进行异步处理,比如今天学习的RxAndroid。

简介

GitHub传送门:https://github.com/ReactiveX/RxAndroid
RxAndroid是一套观察者模式、链式调用的异步编程的API,使用RxAndroid编写的程序在逻辑上将会更加的简介、更加的便于理解。

引入框架

首先

在Androidstudio中的build.gradle(project)文件中的dependencies节点下添加代码
classpath ‘me.tatarka:gradle-retrolambda:3.2.5’

buildscript {repositories {google()jcenter()}dependencies {classpath 'com.android.tools.build:gradle:3.5.0'// NOTE: Do not place your application dependencies here; they belong// in the individual module build.gradle filesclasspath 'me.tatarka:gradle-retrolambda:3.2.5'}
}

第二步

在build.gradle(app)文件下的dependencies节点中添加
implementation ‘io.reactivex.rxjava2:rxandroid:2.1.1’
implementation ‘io.reactivex.rxjava2:rxjava:2.2.12’

dependencies {implementation fileTree(dir: 'libs', include: ['*.jar'])implementation 'androidx.appcompat:appcompat:1.0.2'implementation 'androidx.constraintlayout:constraintlayout:1.1.3'testImplementation 'junit:junit:4.12'androidTestImplementation 'androidx.test:runner:1.1.1'androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1'//rximplementation 'io.reactivex.rxjava2:rxandroid:2.1.1'implementation 'io.reactivex.rxjava2:rxjava:2.2.12'
}

第三步

点击sync now即可
在这里插入图片描述

HelloWord

首先写一个HelloWord,了解一下RxAndroid的简单使用

 public static void helloWord() {//创建被观察者Observable<String> observable = Observable.create(new ObservableOnSubscribe<String>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<String> e) throws Exception {e.onNext("Hello RxAndroid");//发送消息e.onComplete();//完成消息的发送}});//创建观察者Observer observer = new Observer() {@Overridepublic void onSubscribe(@NonNull Disposable d) {Log.e("建立连接", "OnSubscribe");}@Overridepublic void onNext(@NonNull Object o) {//获取被观察者发送过来的消息Log.e("被观察者发送过来的消息", o.toString());}@Overridepublic void onError(@NonNull Throwable e) {Log.e("错误", e.getMessage());}@Overridepublic void onComplete() {Log.e("传输完成", "onComplete");}};//建立连接observable.subscribe(observer);}

在上面的那个方法中,我们可以看到有被观察者:Observable,观察者:Observer,最后通过Observable的subscribe方法订阅连接两者,然后在被观察者中发送一个Hello RxAndroid的字符串,在观察者中的onNext方法中接收到被观察者发送过来的消息。这样一次观察者和被观察者之间的通信就基本完成了。
其中Observer有三个回调方法:onSubscribe、onNext、onError、onComplete,其中onSubscribe是两者在订阅的时候回调,onNext是被观察者调用onNext方法发送消息时的回调,onError是发生异常时的回调,onComplete是被观察者调用onComplete方法时的回调。

-----------------------------------------------说正事专用分割线---------------------------------------------
由以上可以简单的总结一下,RxAndroid有以下三个基本的元素
          1、被观察者(Observable)
          2、观察者(Observer)
          3、订阅(subscribe)

简单使用及常用操作符

我们在HelloWord中在被观察者中向观察者中发送了一个字符串就需要写了这么多,这样是不是太复杂了?不要着急,RxAndroid还为我们提供了一些其他的操作符。

just

通过just操作符创建被观察者,当有一个参数,观察者就会收到一次,两个参数收到两次,以此类推。

 public static void just() {//创建被观察者Observable<String> observable = Observable.just("你好", "222");//创建观察者Consumer<String> consumer = new Consumer<String>() {@Overridepublic void accept(String s) throws Exception {Log.e("通过just操作符创建被观察者", s);}};//建立连接observable.subscribe(consumer);}

有没有发现,通过just创建的被观察者没有onError这个回调,那我应该怎么处理异常呢,这个时候就需要我们手动去抛出异常了,例如;

 public static void justEx() {//创建被观察者try {Observable<Integer> observable = Observable.just(Integer.parseInt("M"));//创建观察者Consumer<Integer> consumer = new Consumer<Integer>() {@Overridepublic void accept(Integer s) throws Exception {Log.e("just操作符", s + "");}};//建立连接observable.subscribe(consumer);} catch (Exception e) {Log.e("justEx", e.getMessage());}}

我在被观察者中发送了一个字符"M",并且将它转换成int类型,因此就会报错,我们可以在catch中打印错误发现: E/justEx: Invalid int: “M”
在这里插入图片描述
我们也可以根据Observer中的三个状态创建回调,通过action代替onComplete

 public static void action() {//创建被观察者Observable<String> observable = Observable.just("你好");//创建观察者//接收onNext发送过来的正常的消息Consumer<String> nextConsumer = new Consumer<String>() {@Overridepublic void accept(String s) throws Exception {Log.e("onNext", s + "");}};//接收异常消息Consumer<Throwable> onErrorConsumer = new Consumer<Throwable>() {@Overridepublic void accept(Throwable throwable) throws Exception {Log.e("异常", "" + throwable.getMessage());}};//创建actionAction completeAction = new Action() {@Overridepublic void run() throws Exception {Log.e("actiong", "任务完成");}};//建立连接observable.subscribe(nextConsumer, onErrorConsumer, completeAction);}

打印log可以看到,依次调用了接收消息、消息完成
在这里插入图片描述

map

变化操作符,对被观察者发送的每一个事件应用一个函数,通过这个函数产生变化

 public static void map() {Observable.create(new ObservableOnSubscribe<String>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<String> e) throws Exception {e.onNext("3");}}).map(new Function<String, Integer>() {@Overridepublic Integer apply(@NonNull String s) throws Exception {return Integer.parseInt(s);}}).subscribe(new Consumer<Integer>() {@Overridepublic void accept(Integer integer) throws Exception {Log.e("返回值", "" + integer);}});}

zip

将多个被观察者发送的事件结合到一起,然后再发送,并且发送事件的数量和上游事件最少的相同,比如:被观察者A发送了两个事件,被观察者B发送了一个事件,那么观察者只能接受一个事件,被观察者的组合也是严格按照顺序来的

public static void zip() {//创建被观察者Observable<String> obs1 = Observable.create(new ObservableOnSubscribe<String>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<String> e) throws Exception {e.onNext("我是1");e.onNext("我是2");e.onComplete();}});Observable<String> obs2 = Observable.create(new ObservableOnSubscribe<String>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<String> e) throws Exception {e.onNext("我是A");e.onComplete();}});//创建观察者Consumer<String> consumer = new Consumer<String>() {@Overridepublic void accept(String o) throws Exception {Log.e("返回值", o);}};//通过zip打包Observable.zip(obs1, obs2, new BiFunction<String, String, String>() {@Overridepublic String apply(@NonNull String s, @NonNull String s2) throws Exception {return s + s2;}}).subscribe(consumer);}

merge

将多个被观察者合并成一个,这里和zip不同,zip是和具有最少的发送事件的被观察者的数量相同

public static void merge() {//创建被观察者Observable<String> obs1 = Observable.create(new ObservableOnSubscribe<String>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<String> e) throws Exception {e.onNext("我是1");e.onNext("我是2");e.onComplete();}});Observable<String> obs2 = Observable.create(new ObservableOnSubscribe<String>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<String> e) throws Exception {e.onNext("我是A");e.onComplete();}});//创建观察者Consumer<String> consumer = new Consumer<String>() {@Overridepublic void accept(String o) throws Exception {Log.e("返回值", o);}};//通过merge打包发送Observable.merge(obs1, obs2).subscribe(consumer);}

filter

操作符用于去除不符合条件的事件。例如下面例子中去掉数组中的空字符串

 public static void filter() {Observable.fromArray(new String[]{"123456789", "", "123456789", "1111"}).filter(new Predicate<String>() {@Overridepublic boolean test(@NonNull String s) throws Exception {//返回false表示去除该事件return s.length() != 0;}}).subscribe(new Consumer<String>() {@Overridepublic void accept(String s) throws Exception {Log.e("接受值", "111" + s);}});}

线程调度

如果说rxandroid只有上面那些功能是远远不会这么火热的,在android中线程间的调度是少不了的。
看一个简单例子

  public static void threadTest() {//创建被观察者Observable<String> observable = Observable.create(new ObservableOnSubscribe<String>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<String> e) throws Exception {Log.e("被观察者", "当前线程:" + Thread.currentThread().getName());e.onNext("测试");e.onError(new Exception("111"));e.onComplete();}});//创建观察者Consumer<String> consumer = new Consumer<String>() {@Overridepublic void accept(String s) throws Exception {Log.e("观察者", "当前线程:" + Thread.currentThread().getName());//  Log.e("消息",""+s);}};//异常Consumer<Throwable> error = new Consumer<Throwable>() {@Overridepublic void accept(Throwable throwable) throws Exception {Log.e("异常", "当前线程:" + Thread.currentThread().getName());}};//任务完成Action action = new Action() {@Overridepublic void run() throws Exception {Log.e("Action", "当前线程:" + Thread.currentThread().getName());}};//建立连接observable.subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).subscribe(consumer, error, action);}

首先创建一个被观察者、观察者,然后在订阅的时候进行指定观察者和被观察者各自的运行线程即可,通过subscribeOn方法指定被观察者运行的线程,Schedulers.io()代表开启一个新线程,一般用于进行访问网络的操作,通过observeOn指定观察者的线程,AndroidSchedulers.mainThread()代表运行的线程在主线程及UI线程,在该线程中可以进行更新UI的操作。

异常处理

如果被观察者中运行的方法抛出异常,观察者则会回调onError方法,并且被观察者中的代码不会继续执行

 public static void onError1() {//创建被观察者Observable<Integer> observable = Observable.create(new ObservableOnSubscribe<Integer>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<Integer> e) throws Exception {e.onNext(2);int i = Integer.parseInt("l");e.onNext(i);int w = Integer.parseInt("e");e.onNext(w);e.onNext(3);Log.e("Observable", "我走到这里啦");e.onComplete();}});//创建观察者Observer<Integer> observer = new Observer<Integer>() {@Overridepublic void onSubscribe(@NonNull Disposable d) {Log.e("建立连接", "onSubscribe");}@Overridepublic void onNext(@NonNull Integer integer) {Log.e("获取返回值", integer + "");}@Overridepublic void onError(@NonNull Throwable e) {Log.e("异常", e.getMessage());}@Overridepublic void onComplete() {Log.e("传输完成", "onComplete");}};//建立连接observable.subscribe(observer);}

被观察中只能调用一次onError方法,如果多次调用,将在第二次调用时抛出异常,并且一旦调用了onError方法,将不能调用onComplete方法报完成

 public static void onError2() {//创建被观察者Observable<String> observable = Observable.create(new ObservableOnSubscribe<String>() {@Overridepublic void subscribe(@NonNull ObservableEmitter<String> e) throws Exception {e.onError(new Exception("错误1"));e.onNext("111");e.onError(new Exception("错误2"));e.onComplete();}});//创建观察者Observer<String> observer = new Observer<String>() {@Overridepublic void onSubscribe(@NonNull Disposable d) {Log.e("建立连接", "onSubscribe");}@Overridepublic void onNext(@NonNull String integer) {Log.e("获取返回值", integer + "");}@Overridepublic void onError(@NonNull Throwable e) {Log.e("异常", e.getMessage());}@Overridepublic void onComplete() {Log.e("传输完成", "onComplete");}};//建立连接observable.subscribe(observer);}

参考链接:
https://www.jianshu.com/p/7eb5ccf5ab1e
https://juejin.im/post/5b17560e6fb9a01e2862246f
https://www.jianshu.com/p/c66951952c74

结语

嗯~,目前我学习到这的也只有这么多了,如果哪里有误还烦请指正呀
点击这里获取找到我,获取更多哦

QQ群:

image.png

微信公众号

分享小知识,记录你的小故事呀
微信公众号.jpg


http://chatgpt.dhexx.cn/article/KQ2Rt4Ml.shtml

相关文章

rxandroid 基础知识

概述 在Android 中, 使用 rxandroid , rxandroid和rxJava的关系是,rxandroid包 依赖rxJava包,在其功能上增加了一些Android特有功能,项目中如果不需要指定rxJava包的版本,只需引入rxandroid包即可,如果需要更改 rxandroid包中默认的rxJava包版本 , 在项目中引入指定的rxJava包即…

RxAndroid使用初探;简洁、优雅、高效

引言 RxAndroid是一个开发库、是一种代码风格、也是一种思维方式。 正如标题所言,RxAndroid的特点是简洁、优雅、高效,它的优点是多线程切换简单、数据变换容易、代码简洁可读性好、第三方支持丰富易于开发;缺点是学习成本较高、出错难以排查。 用途与优势 起源 RxAndroid…

RxAndroid的学习和研究

1.什么是RxAndroid RxAndroid的含义为响应式编程&#xff0c;Rx含义是响应式编程&#xff0c;其本质就是观察者模式&#xff0c;以观察者&#xff08;Observer&#xff09;和订阅者&#xff08;Subscriber&#xff09;为基础的异步响应方式。    Observables发出一系列事件&a…

linux基本功系列之dd命令实战

文章目录 前言&#x1f680;&#x1f680;&#x1f680;一. dd 命令介绍二. 语法格式及常用选项三. 参考案例3.1 创建指定大小的文件3.2 清空磁盘数据3.3 给磁盘做备份还原3.4 把光盘拷贝到root下3.5 内存不足的处理方法 四. 文中出现的概念解释swapon命令介绍4.2 /dev/zero 介…

Linux系统中dd命令用法详解

命令介绍&#xff1a; Linux dd 命令用于读取、转换并输出数据。dd 可从标准输入或文件中读取数据&#xff0c;根据指定的格式来转换数据&#xff0c;再输出到文件、设备或标准输出。 参数介绍 if 代表输入文件。如果不指定 if&#xff0c;默认就会从 stdin 中读取输入。of …

dd 命令详解

dd命令是Linux/Unix下的一个很常见的文件拷贝工具。 我们先列下dd命名的常用的参数&#xff0c;再详细分析&#xff1a; bsBYTES read and write up to BYTES bytes at a time cbsBYTES convert BYTES bytes at a time convCONVS convert the file as pe…

dd命令使用总结

dd命令介绍 dd是Linux下一个非常有用的命令&#xff0c;该命令用于读取、转换并输出数据&#xff1b;dd命令在Android shell下也支持使用。 语法格式&#xff1a; dd [option]dd指令选项详解 iffile&#xff1a;输入文件名&#xff0c;缺省为标准输入 offile&#xff1a;输…

dd命令相关整理

对于一个软件测试人员而言&#xff0c;工作开展前就是准备自己的测试环境&#xff0c;那么重装系统就是首当其冲的一个必备技能。最近因为手边工作环境没有windows的系统&#xff0c;所以没有条件利用软碟通这类刻录软件直接刻录启动盘。被逼无奈之下用命令来刻录&#xff0c;整…

Linux:shell 脚本 自动解压压缩文件tar.gz到指定目录

具体情境 Ubuntu16.04系统&#xff0c;将.tar.gz格式的文件从/home/myftp/upload/nuodongiot目录自动解压到/home/myftp/upload/backupcopy目录中&#xff0c;并将源目录/home/myftp/upload/nuodongiot中的文件移动至/home/myftp/upload/extarct目录中 该过程进行单个文件进行…

tar解压文件至指定目录,不包含原目录

1、tar解压文件至指定目录&#xff0c;不包含原目录 要解压的压缩包原目录结构如下 tar -zxf log.tar.gz --strip-components 1 -C /opt/new_test注&#xff1a; --strip-components 1 解压至下一级目录&#xff0c;若为2则解压至下下级目录 2、压缩只指定的目录&#xff0c…

linux gz解压 指定目,linux解压tar.gz到指定文件夹或目录

1. 前言 本文主要讲解如何解压tar.gz到指定文件夹或目录,tar是Linux系统上的一种打包与压缩工具。 2. linux解压tar文件使用案例 Linux下使用tar命令把当前目录下的zcwyou.tar.gz解压到指定的目录/123/abc/ ,前提要保证存在/123/abc/这个目录。 [root@zcwyou ~]# tar -zxvf zc…

Linux tar 命令 将归档内指定文件解压到指定目录

首先介绍一下 tar 命令&#xff1a; 用途&#xff1a;打包文件&#xff08;制作归档文件&#xff09;、释放归档文件 格式&#xff1a; tar [选项]... 归档文件名 源文件或目录 tar [选项]... 归档文件名 [-C 目标目录] 常用命令选项&#xff1a; -c 创建 .tar 格式…

20191004在LINUX下如何将tar压缩文件解压到指定的目录下

百度搜索&#xff1a;tar 解压缩到指定目录 https://zhidao.baidu.com/question/9844116.html 在LINUX下如何将tar压缩文件解压到指定的目录下 各位&#xff0c;请教一下在LINUX下如何将tar压缩文件解压到指定的目录下&#xff0c;直接用tar xvf 解压出来的是放在当前目录的&am…

关于linux打包以及解压到指定目录的简单操作demo

1.打包到指定目录 命令:tar zcvf /root/test99/a.tar.gz a.txt 1.1打包到当前目录 命令:tar -zcvf a.tar.gz a.txt 2.解压到指定目录 命令: tar -zxvf a.tar.gz -C /root/test99 2.2解压到当前目录 命令:tar -zxvf a.tar.gz 打zip包: 方法如下&#xf…

linux tar解压文件至指定目录,不包含原目录

1、tar解压文件至指定目录&#xff0c;不包含原目录 要解压的压缩包原目录结构如下 通过 --strip-components 1 参数 解压到指定目录或当前目录&#xff08;不含打包前原目录&#xff09; tar zxf log.tar.gz --strip-components 1 -C /opt/new_test注&#xff1a; --strip-co…

Linux拓展之产生随机数

在 Linux 中可以通过内置变量 RANDOM 来产生随机数&#xff0c;该变量会产生一个 [0, 32767] 范围内的随机整数。如下&#xff1a; echo $RANDOM如果要产生 [0-10] 之内的随机整数&#xff1a;echo $(( $RANDOM % 10 )) 如果要产生 [1-10] 之内的随机整数&#xff1a;echo $((…

Linux生成随机数

生成随机数的方法有7种 1.通过时间获取随机数 1&#xff09;date %s &#xff08;随机生成10位数字&#xff09; 用于获得时间戳。 如果用它做随机数&#xff0c;相同一秒的数据是一样的。在做循环处理&#xff0c;多线程里面基本不能满足要求了。 2&#xff09;date…

Linux时间子系统之定时事件层(Clock Events)

几乎所有的计算机系统中都会存在一个所谓的定时设备&#xff0c;经过设置后&#xff0c;在某个固定的时间或某个相对的时间间隔后&#xff0c;达到触发条件&#xff0c;发送中断给处理器。 系统中的每一种实际的定时事件设备都由一个叫做clock_event_device的结构体变量表示&a…

Linux随机数发生器

Linux随机数发生器 日期&#xff1a;2017-11-29 01:42:10 星期三 Linux随机数发生器 一、源代码的基本情况 Linux内核版本涉及文件功能概述 二、外部访问接口 内核层输出接口用户层输出接口环境噪音输入接口 三、核心源码分析 随机数发生器理论熵池结构熵的加入随机数的生成启动…

linux内核随机数函数测试

OS版本&#xff1a;openEuler 22.03 LTS 架构&#xff1a;x86_64 描述&#xff1a;调用内核随机数生成函数&#xff0c;并打印。 // random_num_test.c#include <linux/kernel.h> #include <linux/module.h> #include <linux/random.h>static int __init …