`

最干的RxJava介绍

 
阅读更多

copy from https://www.jianshu.com/p/64aa976a46be

 

一、Obervable的创建

1.create

//通过create创建observable对象,在call中调用subscriber的onnext方法
Observable<String> observable = Observable.create(new Observable.OnSubscribe<String>() {          
  @Override    
  public void call(Subscriber<? super String> subscriber) {              
    for (int i = 0;i < 20;i++)        {            
      subscriber.onNext("fuck i is " + i);        
    }    
    subscriber.onCompleted();
  }});
//上面的代码我们已经构建了一个观察者,我们接下来新建一个订阅者
Subscriber<String> subscriber = new Subscriber<String>()             
{   
   @Override    
    public void onCompleted() {  
          Log.i("rxjava", "onCompleted");       
    }          
    @Override    
    public void onError(Throwable e) {    
     Log.i("rxjava", "error");    
    }    
    @Override    
    public void onNext(String o) {            
      Log.i("rxjava", o);    
}};
//通过调用subscribe方法使观察者和订阅者产生关联,一旦订阅就观察者就开始发送消息
  observable.subscribe(subscriber);

2.from

    //在下面的例子代码中,我们从一个已有的列表中创建一个Observable序列:
    List<String> items = new ArrayList<String>();      
       items.add("1");
       items.add("10");
       items.add("100");
       items.add("200");
    Observable<String> observableString = Observable.from(items);
    //有了observable,再调用1中的subscribe方法即可开始打印
    observable.subscribe(subscriber);

3.just

//通过调用just方法,传入你想发送的数据源,当订阅者进行订阅的时候就开始打印数据
Observable<String> observableString = Observable.just("i", "love", "you","very", "much");
observable.subscribe(subscriber);
//just()方法可以传入一到九个参数,它们会按照传入的参数的顺序来发射它们。just()方法也可以接受列表或数组,就像from()方法,但是它不会迭代列表发射每个值,它将会发射整个列表。通常,当我们想发射一组已经定义好的值时会用到它。

4、repeat重复

//假如你想对一个Observable重复发射三次数据。例如,我们用just()例子中的Observable:
Observable<String> observableString = Observable.just("i", "love", "you","very", "much").repeat(3);
observable.subscribe(subscriber);//通过添加repeat(3),just里面的内容会被打印3次

5、range

从一个指定的数字x开发发射n个数字
Observable.range(10,3)
    .subscribe(new Observer<Integer>() {

        @Override
        public void onCompleted() {
            Toast.makeText(getActivity(), "Yeaaah!", Toast.LENGTH_LONG).show();
        }

        @Override
        public void onError(Throwable e) {
            Toast.makeText(getActivity(), "Something went wrong!",Toast.LENGTH_SHORT).show();
        }

        @Override
        public void onNext(Integer number) {
            Toast.makeText(getActivity(), "I say " + number, Toast.LENGTH_SHORT).show();
        }
    });
    //range()函数用两个数字作为参数:第一个是起始点,第二个是我们想发射数字的个数。

6、interval轮询

//interval()函数在你需要创建一个轮询程序时非常好用。
 Observable.interval(3,TimeUnit.SECONDS)
    .subscribe(new Observer<Integer>() {

        @Override
        public void onCompleted() {
            Toast.makeText(getActivity(), "Yeaaah!", Toast.LENGTH_LONG).show();
        }

        @Override
        public void onError(Throwable e) {
            Toast.makeText(getActivity(), "Something went wrong!", Toast.LENGTH_SHORT).show();
        }

        @Override
        public void onNext(Integer number) {
            Toast.makeText(getActivity(), "I say " + number, Toast.LENGTH_SHORT).show();
        }
    });
//interval()函数的两个参数:一个指定两次发射的时间间隔,另一个是用到的时间单位。这个只会执行一次,添加subscribeOn(Schedulers.newThread())好像能够达到轮训的效果

7、timer

//如果你需要一个一段时间之后才发射的Observable,你可以像下面的例子使用timer():
Observable.timer(3,TimeUnit.SECONDS)
    .subscribe(new Observer<Long>() {

        @Override
        public void onCompleted() {

        }

        @Override
        public void onError(Throwable e) {

        }

        @Override
        public void onNext(Long number) {
            Log.d("RXJAVA", "I say " + number);
        }
    });
//它将3秒后发射0,然后就完成了。

二、过滤

这一章中,我们将研究可观测序列的本质:过滤。我们将学到如何从发射的Observable中选取我们想要的值,如何获取有限个数的值,如何处理溢出的场景,以及更多的有用的技巧。

1、filter

//RxJava让我们使用filter()方法来过滤我们观测序列中不想要的值
Observable.just("H1", "h2","h3","h4","h5").filter(new Func1<String, Boolean>() {
            @Override
            public Boolean call(String s) {
                return s.startsWith("H");
            }
        }).subscribe(new Subscriber<String>() {
            @Override
            public void onCompleted() {

            }

            @Override
            public void onError(Throwable e) {

            }

            @Override
            public void onNext(String s) {
                Log.i(tag, s);
            }
        });
//通过设置filter,然后在call里面添加s.startsWith("H"),如果是H开头就返回true,否则返回false。从而能够过滤掉不是h开头的消息,打印出以H开头的消息。

2、take

//take()函数用整数N来作为一个参数,从原始的序列中发射前N个元素,然后完成:
Observable.just("H1", "h2","h3","h4","h5").take(3).subscribe(new Subscriber<String>() {
            @Override
            public void onCompleted() {

            }

            @Override
            public void onError(Throwable e) {

            }

            @Override
            public void onNext(String s) {
                Log.i(tag, s);
            }
        });
//通过设置take(3),就可以取出前3个消息,打印出H1,h2,h3
注:如果想从后面取数据,可以调用takeLast(3)取最后3条消息

3、Distinct

//我们可以对我们的序列使用distinct()函数去掉重复的。就像takeLast()一样,distinct()作用于一个完整的序列,然后得到重复的过滤项,它需要记录每一个发射的值。
Observable o = Observable.just("H1", "h2","h3","h4","h5").repeat(3);//输入重复3遍
o.distinct().subscribe(new Subscriber<String>() {
            @Override
            public void onCompleted() {

            }

            @Override
            public void onError(Throwable e) {

            }

            @Override
            public void onNext(String s) {
                Log.i(tag, s);
            }
        });

4、DistinctUntilsChanged

//它能轻易的忽略掉所有的重复并且只发射出新的值。
//下图用图形化的方式展示了我们如何将distinctUntilChanged()函数应用在一个存在的序列上来创建一个新的不重复发射元素的序列。
 
 

5、First and last

first()方法和last()方法很容易弄明白。它们从Observable中只发射第一个元素或者最后一个元素。

5.1、first

 
 

5.2、last

 

6、skip和skipLast

skip()skipLast()函数与take()takeLast()相对应。它们用整数N作参数,从本质上来说,它们不让Observable发射前N个或者后N个值。

 
skip(2)
 
 

举个例子

Observable.just("hello", "my", "world").skip(1).skipLast(1)
                .subscribe(new Subscriber<String>() {
                    @Override
                    public void onCompleted() {

                    }

                    @Override
                    public void onError(Throwable e) {

                    }

                    @Override
                    public void onNext(String s) {
                        Log.i(tag, s);
                    }
                });
//跳过第一个hello和最后一个world,打印my

7、ElementAt和elementAtOrDefault

elementAt()函数仅从一个序列中发射第n个元素然后就完成了。如果我们想查找第五个元素但是可观测序列只有三个元素可供发射时该怎么办?我们可以使用elementAtOrDefault()

下图展示了如何通过使用elementAt(2)从一个序列中选择第三个元素以及如何创建一个只发射指定元素的新的Observable。

 
 

举个例子

        Observable.just("hello", "my", "world").elementAt(1)
                .subscribe();
        Observable.just("hello", "my", "world").elementAtOrDefault(10, "null")
                .subscribe();

8、Sampling

在Observable后面加一个sample(),我们将创建一个新的可观测序列,它将在一个指定的时间间隔里由Observable发射最近一次的数值:

 
 
List<String> list = new ArrayList<>();
        for (int i = 0; i < 1000; i++) {
            list.add("i = " + i);
        }
        Observable<String> observable = Observable.from(list);
        observable.sample(50, TimeUnit.MILLISECONDS).subscribe(new Subscriber<String>() {
            @Override
            public void onCompleted() {

            }

            @Override
            public void onError(Throwable e) {

            }

            @Override
            public void onNext(String s) {
                Log.i(tag, s);
            }
        });
        //50毫秒取一次最近的消息进行打印

9、timeout

我们可以认为timeout()为一个Observable的限时的副本。如果在指定的时间间隔内Observable不发射值的话,它监听的原始的Observable时就会触发onError()函数。我们可以使用timeout()函数来监听源可观测序列,就是在我们设定的时间间隔内如果没有得到一个值则发射一个错误。

 
 

三、Observable变化

3.1 map家族

RxJava提供了几个mapping函数:map(),flatMap(),concatMap(),flatMapIterable()以及switchMap().所有这些函数都作用于一个可观测序列,然后变换它发射的值,最后用一种新的形式返回它们。让我们用合适的“真实世界”的例子一个个的学习下。

3.1.1 map

RxJava的map函数接收一个指定的Func对象然后将它应用到每一个由Observable发射的值上。下图展示了如何将一个乘法函数应用到每个发出的值上以此创建一个新的Observable来发射转换的数据。

 
 

举个例子:接收一系列的数字,经过map转成string类型,然后打印出来。

Observable.just(1, 2, 3, 4, 5)
                .map(new Func1<Integer, String>() {
                    @Override
                    public String call(Integer integer) {
                        return "the position is" + integer;
                    }
                }).subscribe(new Observer<String>() {
            @Override
            public void onCompleted() {

            }

            @Override
            public void onError(Throwable e) {

            }

            @Override
            public void onNext(String s) {
                Log.i(tag, s);
            }
        });

3.1.2 flatMap

flatMap是map中比较常见但也是比较难理解内。

在复杂的场景中,我们有一个这样的Observable:它发射一个数据序列,这些数据本身也可以发射Observable。RxJava的flatMap()函数提供一种铺平序列的方式,然后合并这些Observables发射的数据,最后将合并后的结果作为最终的Observable。

 
 

举个例子:假设我有个天气客户端,某个页面需要获取四个城市(南昌、北京、天津和深圳)的天气。那么首先我们需要传入四个城市名,然后分别根据城市名获取天气信息,然后更行ui更新。

public void flatMap() {
        Observable.just("南昌", "深圳", "天津", "北京").flatMap(new Func1<String, Observable<WeatherInfo>>() {
            @Override
            public Observable<WeatherInfo> call(String s) {
                return getWeather(s);
            }
        }).observeOn(AndroidSchedulers.mainThread()).//更新ui一定要加这句
        subscribe(new Subscriber<WeatherInfo>() {
            @Override
            public void onCompleted() {

            }

            @Override
            public void onError(Throwable e) {

            }

            @Override
            public void onNext(WeatherInfo weatherInfo) {
                //更新ui
            }
        });
    }

    //新建天气信息类
    class WeatherInfo {

    }

    private Observable<WeatherInfo> getWeather(final String city) {
        Observable<WeatherInfo> observable = Observable.create(new Observable.OnSubscribe<WeatherInfo>() {
            @Override
            public void call(Subscriber<? super WeatherInfo> subscriber) {
                subscriber.onNext(getWeatherInfo(city));
                subscriber.onCompleted();
            }
        }).subscribeOn(Schedulers.io());//网络请求一定要加这句
        return observable;
    }

    private WeatherInfo getWeatherInfo(String city) {
        //模拟网络请求返回weatherinfo
        return new WeatherInfo();
    }
    //南昌->能够获取南昌天气的observable->更新ui

3.1.3 ConcatMap

RxJava的concatMap()函数解决了flatMap()的交叉问题,提供了一种能够把发射的值连续在一起的铺平函数,而不是合并它们,如下图所示:

 

注:拿上面的例子来说我们传入的是"南昌", "深圳", "天津", "北京",flatmap转换之后得到的顺序并不是这样,而是被打乱了的;用concaatmap的话就是按顺序的转换,只有这种区别。

3.1.4 SwitchMap

如下图所示,switchMap()flatMap()很像,除了一点:每当源Observable发射一个新的数据项(Observable)时,它将取消订阅并停止监视之前那个数据项产生的Observable,并开始监视当前发射的这一个。

 
 

例子参考

 

SwitchMap

3.1.5 Scan

scan操作符对一个序列的数据应用一个函数,并将这个函数的结果发射出去作为下个数据应用这个函数时候的第一个参数使用,有点类似于递归操作

 
 

举个例子

private Observable<Integer> scanObserver() {
        return Observable.from(list).scan((x, y) -> x * y).observeOn(AndroidSchedulers.mainThread());
    }

结果为:

 
 

3.2 GroupBy

groupBy操作符是对源Observable产生的结果进行分组,形成一个类型为GroupedObservable的结果集,GroupedObservable中存在一个方法为getKey(),可以通过该方法获取结果集的Key值(类似于HashMap的key)。

值得注意的是,由于结果集中的GroupedObservable是把分组结果缓存起来,如果对每一个GroupedObservable不进行处理(既不订阅执行也不对其进行别的操作符运算),就有可能出现内存泄露。因此,如果你对某个GroupedObservable不进行处理,最好是对其使用操作符take(0)处理。

groupBy操作符的流程图如下:

 
 
调用例子如下:
Observable.interval(1, TimeUnit.SECONDS).take(10).groupBy(new Func1<Long, Long>() {
            @Override
            public Long call(Long value) {
                //按照key为0,1,2分为3组
                return value % 3;
            }
        }).subscribe(new Action1<GroupedObservable<Long, Long>>() {
            @Override
            public void call(GroupedObservable<Long, Long> result) {
                result.subscribe(new Action1<Long>() {
                    @Override
                    public void call(Long value) {
                        System.out.println("key:" + result.getKey() +", value:" + value);
                    }
                });
            }
        });
运行结果如下: 
key:0, value:0 
key:1, value:1 
key:2, value:2 
key:0, value:3 
key:1, value:4 
key:2, value:5 
key:0, value:6 
key:1, value:7 
key:2, value:8 
key:0, value:9

3.3 Buffer

RxJava中的buffer()函数将源Observable变换一个新的Observable,这个新的Observable每次发射一组列表值而不是一个一个发射。

 
 

上图中展示了buffer()如何将count作为一个参数来指定有多少数据项被包在发射的列表中。实际上,buffer()函数有几种变体。其中有一个是允许你指定一个skip值:此后每skip项数据,然后又用count项数据填充缓冲区。如下图所示:

 
 

举个例子

List<String> list = new ArrayList<>();
for (int i = 0; i < 30; i++)
{
    list.add("hello i:" + i);
}

Observable.from(list).buffer(4).subscribe(new Subscriber<List<String>>() {
    @Override
    public void onCompleted() {

    }

    @Override
    public void onError(Throwable e) {

    }

    @Override
    public void onNext(List<String> strings) {
        for (String s : strings)
        {
            Log.i(tag, s);
        }
        Log.i(tag, "\n next group");
    }
});

3.4 window

RxJava的window()函数和buffer()很像,但是它发射的是Observable而不是列表。下图展示了window()如何缓存3个数据项并把它们作为一个新的Observable发射出去。

 
 
List<String> list = new ArrayList<>();
for (int i = 0; i < 30; i++) {
    list.add("hello i:" + i);
}
Observable.from(list).window(4).subscribe(new Subscriber<Observable<String>>() {
    @Override
    public void onCompleted() {

    }

    @Override
    public void onError(Throwable e) {

    }

    @Override
    public void onNext(Observable<String> stringObservable) {
        stringObservable.subscribe(new Subscriber<String>() {
            @Override
            public void onCompleted() {

            }

            @Override
            public void onError(Throwable e) {

            }

            @Override
            public void onNext(String s) {
                Log.i(tag, s);
            }
        });
        Log.i(tag, "\n next group");
    }
});

这些Observables中的每一个都发射原始Observable数据的一个子集,数量由count指定,最后发射一个onCompleted()结束。正如buffer()一样,window()也有一个skip变体,如下图所示:

 
 

3.5 cast

RxJava的cast()函数是本章中最后一个操作符。它是map()操作符的特殊版本。它将源Observable中的每一项数据都转换为新的类型,把它变成了不同的Class

 
 
public void cast() {
    List<Person> list = new ArrayList<>();
    for (int i = 0; i < 30; i++) {
        list.add(new Male(true, "i" + i));
    }
    Observable.from(list).cast(Male.class).subscribe(new Subscriber<Male>() {
        @Override
        public void onCompleted() {

        }

        @Override
        public void onError(Throwable e) {

        }

        @Override
        public void onNext(Male s) {
            Log.i(tag, "name:" + s.name + "\n hasJJ:" + s.hasJJ);
        }
    });
}

class Person {
    String name;

    public Person(String name) {
        this.name = name;
    }
}

class Male extends Person {
    public boolean hasJJ;

    public Male(boolean hasJJ, String name) {
        super(name);
        this.hasJJ = hasJJ;
    }
}
//多态,我们用Person接收Male对象,然后通过cast转成Male类型。有点instanceof的意思。

四、组合Observables

我们学到如何转换可观测序列。我们也看到了map(),scan(),groupBY(),以及更多有用的函数的实际例子,它们帮助我们操作Observable来创建我们想要的Observable。

我们将研究组合函数并学习如何同时处理多个Observables来创建我们想要的Observable。

4.1、merge

RxJava的merge()方法将帮助你把两个甚至更多的Observables合并到他们发射的数据项里。下图给出了把两个序列合并在一个最终发射的Observable。

 
 

举个例子

List<String> list1 = new ArrayList<>();
for (int i = 0; i < 30; i++) {
    list1.add("hello i:" + i);
}
List<String> list = new ArrayList<>();
for (int j = 0; j < 30; j++) {
    list.add("world j:" + j);
}
List<String> list2 = new ArrayList<>();
for (int m = 0; m < 30; m++) {
    list2.add("fuck m:" + m);
}
Observable<String> world = Observable.from(list);
Observable<String> hello = Observable.from(list1);
Observable<String> fuck = Observable.from(list2);
Observable.merge(world, hello, fuck).subscribe(new Subscriber<String>() {
    @Override
    public void onCompleted() {

    }

    @Override
    public void onError(Throwable e) {

    }

    @Override
    public void onNext(String s) {
        Log.i(tag, s);
    }
});

注意错误时的toast消息,你可以认为每个Observable抛出的错误都将会打断合并。如果你需要避免这种情况,RxJava提供了mergeDelayError(),它能从一个Observable中继续发射数据即便是其中有一个抛出了错误。当所有的Observables都完成时,mergeDelayError()将会发射onError(),如下图所示:

 
 

4.2 zip

我们在处理多源时可能会带来这样一种场景:多从个Observables接收数据,处理它们,然后将它们合并成一个新的可观测序列来使用。RxJava有一个特殊的方法可以完成: zip() 合并两个或者多个Observables发射出的数据项,根据指定的函数Func* 变换它们,并发射一个新值。下图展示了 zip() 方法如何处理发射的“numbers”和“letters”然后将它们合并一个新的数据项:

 
 

举个例子

List<String> hellos = new ArrayList<>();
for (int i = 0; i < 30; i++) {
    hellos.add("hello i:" + i);
}
List<Integer> worlds = new ArrayList<>();
for (int j = 0; j < 20; j++) {
    worlds.add(j);
}

Observable.zip(Observable.from(hellos).subscribeOn(Schedulers.io()), Observable.from(worlds).subscribeOn(Schedulers.io()), new Func2<String, Integer, String>() {
    @Override
    public String call(String s, Integer integer) {
        return "index:" + integer + "\t s:" + s;
    }
}).subscribe(new Subscriber<String>() {
    @Override
    public void onCompleted() {
        Log.i(tag, "onCompleted");
    }

    @Override
    public void onError(Throwable e) {
        Log.i(tag, "error");
    }

    @Override
    public void onNext(String s) {
        Log.i(tag, s);
    }
});
//这里只会打印20条,而不是30条,应该是有个observable完成就完成

五、Schedulers

我们提升标准看看如何使用RxJava的调度器来处理多线程和并发编程的问题。我们将学习到如何以响应式的方式创建网络操作,内存访问,以及耗时任务。

5.1、StrictMode

为了获得更多出现在代码中的关于公共问题的信息,我们激活了StrictMode模式。

StrictMode帮助我们侦测敏感的活动,如我们无意的在主线程执行磁盘访问或者网络调用。正如你所知道的,在主线程执行繁重的或者长时的任务是不可取的。因为Android应用的主线程时UI线程,它被用来处理和UI相关的操作:这也是获得更平滑的动画体验和响应式App的唯一方法。

为了在我们的App中激活StrictMode,我们只需要在MainActivity中添加几行代码,即onCreate()方法中这样:

@Override
public void onCreate() { 
    super.onCreate();
    if (BuildConfig.DEBUG) {
        StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder().detectAll().penaltyLog().build()); 
        StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder().detectAll().penaltyLog().build());
    } 
}
//我们并不想它总是激活着,因此我们只在debug构建时使用。这种配置将报告每一种关于主线程用法的违规做法,并且这些做法都可能与内存泄露有关:Activities、BroadcastReceivers、Sqlite等对象。
//选择了penaltyLog(),当违规做法发生时,StrictMode将会在logcat打印一条信息。

阻塞I/O的操作会导致App必须等待结果返回(阻塞结束)才能进行下一步操作。在UI线程上执行一个阻塞操作会将UI强行卡住,直接造成很糟糕的用户体验。

我们激活StrictMode后,我们开始收到了关于我们的App错误操作磁盘I/O的不良信息。

D/StrictMode  StrictMode policy violation; ~duration=998 ms: android.os.StrictMode$StrictModeDiskReadViolation: policy=31 violation=2
at android.os.StrictMode$AndroidBlockGuardPolicy.onReadFromDisk (StrictMode.java:1135)
at libcore.io.BlockGuardOs.open(BlockGuardOs.java:106) at libcore.io.IoBridge.open(IoBridge.java:393)
at java.io.FileOutputStream.<init>(FileOutputStream.java:88) 
at android.app.ContextImpl.openFileOutput(ContextImpl.java:918) 
at android.content.ContextWrapper.openFileOutput(ContextWrapper. java:185)
at com.packtpub.apps.rxjava_essentials.Utils.storeBitmap (Utils.java:30)

上一条信息告诉我们Utils.storeBitmap()函数执行完耗时998ms:在UI线程上近1秒的不必要的工作和App上近1秒不必要的迟钝。这是因为我们以阻塞的方式访问磁盘。我们的storeBitmap()函数包含了:

FileOutputStream fOut = context.openFileOutput(filename, Context.MODE_PRIVATE);

它直接访问智能手机的固态存储然后就慢了。我们该如何提高访问速度呢?storeBitmap()函数保存了已安装App的图标。他的返回值类型为void,因此在执行下一个操作前我们毫无理由去等待直到它完成。我们可以启动它并让它执行在不同的线程。近几年来Android的线程管理发生了许多变化,导致App出现诡异的行为。我们可以使用AsyncTask,但是我们要避免掉入前几章里的onPre... onPost...doInBackGround地狱。下面我们将换用RxJava的方式。调度器万岁!

5.2 、Schedulers

调度器以一种最简单的方式将多线程用在你的Apps的中。它们是RxJava重要的一部分并能很好地与Observables协同工作。它们无需处理实现、同步、线程、平台限制、平台变化而可以提供一种灵活的方式来创建并发程序。

RxJava提供了5种调度器:

  • .io()
  • .computation()
  • .immediate()
  • .newThread()
  • .trampoline()

让我们一个一个的来看下它们:

5.2.1、Schedulers.io

这个调度器时用于I/O操作。它基于根据需要,增长或缩减来自适应的线程池。我们将使用它来修复我们之前看到的StrictMode违规做法。由于它专用于I/O操作,所以并不是RxJava的默认方法;正确的使用它是由开发者决定的。

重点需要注意的是线程池是无限制的,大量的I/O调度操作将创建许多个线程并占用内存。一如既往的是,我们需要在性能和简捷两者之间找到一个有效的平衡点。

5.2.2、Schedulers.computation

这个是计算工作默认的调度器,它与I/O操作无关。它也是许多RxJava方法的默认调度器:buffer(),debounce(),delay(),interval(),sample(),skip()

5.2.3、Schedulers.immediate

这个调度器允许你立即在当前线程执行你指定的工作。它是timeout(),timeInterval(),以及timestamp()方法默认的调度器。

5.2.4、Schedulers.newThread()

这个调度器正如它所看起来的那样:它为指定任务启动一个新的线程。

5.2.5、Schedulers.trampoline

当我们想在当前线程执行一个任务时,并不是立即,我们可以用.trampoline()将它入队。这个调度器将会处理它的队列并且按序运行队列中每一个任务。它是repeat()retry()方法默认的调度器。

5.2.5、SubscribeOn and ObserveOn

我们如何利用它来和Observables一起工作呢?RxJava提供了subscribeOn()方法来用于每个Observable对象。subscribeOn()方法用Scheduler来作为参数并在这个Scheduler上执行Observable调用。

在“真实世界”这个例子中,我们调整loadList()函数。首先,我们需要一个新的getApps()方法来检索已安装的应用列表:

private Observable<AppInfo> getApps() { 
    return Observable.create(subscriber -> {
        List<AppInfo> apps = new ArrayList<>();
        SharedPreferences sharedPref = getActivity().getPreferences(Context.MODE_PRIVATE);
        Type appInfoType = new TypeToken<List<AppInfo>>(){}.getType();
        String serializedApps = sharedPref.getString("APPS", "");
        if (!"".equals(serializedApps)) {
            apps = new Gson().fromJson(serializedApps,appInfoType); 
        }
        for (AppInfo app : apps) {
            subscriber.onNext(app);
        }
        subscriber.onCompleted(); 
    });
}

getApps()方法返回一个AppInfo的Observable。它先从Android的SharePreferences读取到已安装的应用程序列表。反序列化,并一个接一个的发射AppInfo数据。使用新的方法来检索列表,loadList()函数改成下面这样:

private void loadList() {
    mRecyclerView.setVisibility(View.VISIBLE);
    getApps().subscribe(new Observer<AppInfo>() {
        @Override
        public void onCompleted() {
            mSwipeRefreshLayout.setRefreshing(false);
            Toast.makeText(getActivity(), "Here is the list!", Toast.LENGTH_LONG).show();
        }

        @Override
        public void onError(Throwable e) {
            Toast.makeText(getActivity(), "Something went wrong!", Toast.LENGTH_SHORT).show();
            mSwipeRefreshLayout.setRefreshing(false);
        }

        @Override
        public void onNext(AppInfo appInfo) {
            mAddedApps.add(appInfo);
                mAdapter.addApplication(mAddedApps.size() - 1, appInfo);
        } 
    });
}

如果我们运行代码,StrictMode将会报告一个不合规操作,这是因为SharePreferences会减慢I/O操作。我们所需要做的是指定getApps()需要在调度器上执行:

getApps().subscribeOn(Schedulers.io())
        .subscribe(new Observer<AppInfo>() { [...]

Schedulers.io()将会去掉StrictMode的不合规操作,但是我们的App现在崩溃了是因为:

at rx.internal.schedulers.ScheduledAction.run(ScheduledAction.jav a:58)
at java.util.concurrent.Executors$RunnableAdapter.call(Executors. java:422)
at java.util.concurrent.FutureTask.run(FutureTask.java:237) 
at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutu reTask.access$201(ScheduledThreadPoolExecutor.java:152)
at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutu reTask.run(ScheduledThreadPoolExecutor.java:265)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolEx ecutor.java:1112)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolE xecutor.java:587)
at java.lang.Thread.run(Thread.java:841) Caused by:
    android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.

Only the original thread that created a view hierarchy can touch its views.

我们再次回到Android的世界。这条信息简单的告诉我们我们试图在一个非UI线程来修改UI操作。意思是我们需要在I/O调度器上执行我们的代码。因此我们需要和I/O调度器一起执行代码,但是当结果返回时我们需要在UI线程上操作。RxJava让你能够订阅一个指定的调度器并观察它。我们只需在loadList()函数添加几行代码,那么每一项就都准备好了:

getApps()
.onBackpressureBuffer()
.subscribeOn(Schedulers.io())//获取app信息的操作放在io线程
.observeOn(AndroidSchedulers.mainThread())//更新和显示应用的操作放在android的mainThread线程(ui线程)
.subscribe(new Observer<AppInfo>() { [...]

observeOn()方法将会在指定的调度器上返回结果:如例子中的UI线程。onBackpressureBuffer()方法将告诉Observable发射的数据如果比观察者消费的数据要更快的话,它必须把它们存储在缓存中并提供一个合适的时间给它们。做完这些工作之后,如果我们运行App,就会出现已安装的程序列表。

总结:

RxJava为此提供了极其实用的工具:调度器。调度器以及不同应用场景下的优化方案一起,将我们从StrictMode中的不合法操作以及阻塞I/O的方法中解放出来。我们现在可以用简单的,响应式的,并在整个App中保持一致的方式来访问本地存储和网络。

 



作者:安静的学点东西
链接:https://www.jianshu.com/p/64aa976a46be
來源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。
分享到:
评论

相关推荐

    Android代码-一个集Gank.Io,Rxjava示例,操作符,MD控件使用,各种好玩Ap示例的学习App。

    功能介绍 每日干货查看,一键分享保存,分类干货查看,干货搜索. Rxjava操作符,中文文档,使用案例,方便随时查阅学习. 安卓艺术开发探索读书笔记,EffectiveJava读书笔记. 收集各大神博客以及安卓笔记,安卓面试笔记等...

    Android代码-干果-简洁易用的 gank 客户端

    介绍 一款基于GankIo开发的练手项目。项目采用 MVP Dagger2 Retrofit RxJava开发。首页布局样式大致参考掘金客户端。 应用截图 Gif 特性 基本遵循MD风格 MVP dagger2架构模式的项目应用 retrofit rxjava配合...

    RxJavaDemo:本教程基于 RxJava2 实践练习 Demo

    RxJavaDemo RxJava+Retrofit2 的常见实际使用场景 本项目是模仿 在模仿了几个RxJava+Retrofit2的项目之后,可能缺少没有系统学习,总感觉还不能掌握 RxJava+Retrofit2 等开源库的精髓,结合了在网上...或许是介绍Android S

    AndroidRank 干货学习客户端

    功能介绍 每日干货查看,一键分享保存,分类干货查看,干货搜索. Rxjava 操作符,中文文档,使用案例,方便随时查阅学习. 安卓艺术开发探索读书笔记,EffectiveJava 读书笔记. 收集各大神博客以及安卓笔记,安卓面试笔记等...

    Android代码-LGank

    介绍 整体设计遵循 Material Design 规范,实现了多种应用常见的效果。包括不局限于: 沉浸式状态栏 有趣的加载动画 滑动关闭Activity 多主题切换 瀑布流、线性、网格布局动态切换 动态加载流式布局 下拉刷新、上拉...

    安卓mvvm纯干货

    ViewModel不直接与View或Model交互,而是通过数据绑定实现与View的同步,同时通过LiveData或RxJava等观察者模式与Model进行通信。 二、MVVM的优势 1. 分离关注点:MVVM模式将数据处理和UI更新分离,使得代码结构...

    Android-用知乎和gankio网易新闻豆瓣电影的API

    6. **API使用**:项目中的知乎、Gank.io、网易新闻和豆瓣电影API提供了各种信息,如知乎的热门话题、Gank.io的技术干货、网易新闻的实时资讯以及豆瓣电影的评分和介绍等。开发者需要理解这些API的接口文档,正确构造...

    gank毕业设计—(包含完整源码可运行).zip

    5. `README.md`: 项目说明文档,一般会介绍项目的目的、功能及如何运行。 二、技术栈解析 这个毕业设计项目可能采用了以下技术: 1. Android Studio:作为主要的开发环境,提供了强大的集成开发环境。 2. MVVM...

    CloudReader:云阅览:一种基于网易云音乐UI,使用WanAndroid,Gank.Io Api,MVVM-DataBinding架构开发的符合Google Material Design的Android客户端

    CloudReader Kotlin &&网易云音乐Ui && Retrofit2 + RxJava2 + MVVM-databinding && Wanandroid,Gank.Io Api介绍一款基于网易云音乐UI,使用WanAndroid,Gank.Io Api,MVVM-DataBinding架构开发的符合Google ...

    Reactor指南中文版 2.0

    在介绍Reactor之前,我们首先要知道何为异步编程。异步编程指的是程序的某部分能够在等待其他部分完成时继续执行其他任务。在处理大量数据或需要快速响应的场景中,异步编程尤其重要,可以极大地提升应用的性能和...

Global site tag (gtag.js) - Google Analytics