6章 RxJava基础实战

本篇文章已授权微信公众号 YYGeeker 独家发布转载请标明出处

CSDN学院课程地址

6. RxJava基础实战

6.1 模拟发送验证码

应用场景:当用户点击发送验证码后,在倒计时的时间内是不可以重新点击发送验证码的,倒计时结束后,发送验证码的按钮重新恢复点击,这里举例子为3s的倒计时

public void verify(View view) {
    final long count = 3;//倒计时时间
    final Button button = (Button) view;//当前按钮

    Observable.interval(0, 1, TimeUnit.SECONDS)//定时器
            .take(count + 1)//取定时器前4个,当前值:0,1,2,3
            .map(new Function<Long, Long>() {
                @Override
                public Long apply(@NonNull Long aLong) throws Exception {
                    return count - aLong;//将值转换下,当前值:3,2,1,0
                }
            })
            .observeOn(AndroidSchedulers.mainThread())//主线程更新UI
            .doOnSubscribe(new Consumer<Disposable>() {
                @Override
                public void accept(@NonNull Disposable disposable) throws Exception {
                    //监听订阅时,将按钮设置为不可点击
                    button.setEnabled(false);
                    button.setTextColor(Color.BLACK);
                }
            })
            .subscribe(new Observer<Long>() {
                @Override
                public void onSubscribe(Disposable d) {}
                @Override
                public void onNext(Long aLong) {
                    //设置倒计时文本
                    button.setText("剩余" + aLong + "秒");
                }
                @Override
                public void onError(Throwable e) {}
                @Override
                public void onComplete() {
                    //事件完成后恢复点击
                    button.setEnabled(true);
                    button.setText("发送验证码");
                }
            });
}

6.2 模拟用户点击防抖动

应用场景:在某些应用场景中,用户会多次点击同一个按钮,导致有多次点击事件的产生,如果点击事件中是网络请求,那么就会产生多次网络请求。正确的操作应该是,在一定时间内,用户频繁点击多次按钮之后,只访问一次网络请求。下面针对所说的需求进行编写

写法一:

/**
 * 模拟用户点击防抖动
 */
public void query(View view) {
    RxUtils.click(view, 2)
            .subscribe(new Observer<Object>() {
                @Override
                public void onSubscribe(Disposable d) {

                }

                @Override
                public void onNext(Object o) {
                    System.out.println("onNext");
                }

                @Override
                public void onError(Throwable e) {

                }

                @Override
                public void onComplete() {
                    System.out.println("onComplete");
                }
            });
}

//封装工具
static class RxUtils {
    static Observable<Object> click(final View view, long seconds) {
        return new ViewClickObservable(view)
                .throttleFirst(seconds, TimeUnit.SECONDS)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .doOnDispose(new Action() {
                                 @Override
                                 public void run() throws Exception {
                                     if (view != null) {
                                         view.setOnClickListener(null);
                                     }
                                 }
                             }
                );
    }
}

//创建一个观察者
static class ViewClickObservable extends Observable<Object> {

    private View view;

    public ViewClickObservable(View view) {
        this.view = view;
    }

    //当这个观察者被订阅的时候,会执行下面的回调
    @Override
    protected void subscribeActual(final Observer<? super Object> observer) {
        if (view != null) {
            view.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    observer.onNext(v);
                }
            });
        }
    }
}

写法二:


static class ViewClickObservableOnSubscribe implements ObservableOnSubscribe<Object> {

    private ObservableEmitter<Object> emitter;

    public ObservableEmitter<Object> getEmitter() {
        return emitter;
    }

    @Override
    public void subscribe(ObservableEmitter<Object> e) throws Exception {
        this.emitter = e;
    }
}

//封装工具
static class RxUtils {

    static Observable<Object> clicks(final View view, long seconds) {
        final ViewClickObservableOnSubscribe viewClickObservableOnSubscribe = new ViewClickObservableOnSubscribe();

        if (view != null) {
            view.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    ObservableEmitter<Object> emitter = viewClickObservableOnSubscribe.getEmitter();
                    if (emitter != null && !emitter.isDisposed()) {
                        emitter.onNext(1);
                    }
                }
            });
        }

        return Observable
                .create(viewClickObservableOnSubscribe)
                .throttleFirst(seconds, TimeUnit.SECONDS)
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .doOnDispose(new Action() {
                                 @Override
                                 public void run() throws Exception {
                                     if (view != null) {
                                         view.setOnClickListener(null);
                                     }
                                 }
                             }
                );
    }
}

/**
 * 模拟用户点击防抖动
 */
public void query(View view) {
    RxUtils.clicks(view, 2)
            .subscribe(new Observer<Object>() {
                @Override
                public void onSubscribe(Disposable d) {

                }

                @Override
                public void onNext(Object o) {
                    System.out.println("onNext");
                }

                @Override
                public void onError(Throwable e) {

                }

                @Override
                public void onComplete() {
                    System.out.println("onComplete");
                }
            });
}

6.3 模拟会员信息的合并

应用场景:假如我们当前在好友聊天列表界面中,客户端需要通过好友列表的uid去查询好友列表中会员信息,然后显示会员图标等信息。假如聊天列表界面中的好友数量有成千上百个,客户端每次从后台批量查询用户会员信息后,需要在本地做缓存,如果这时用户在聊天列表中进入某个群聊界面,这个时候还是需要去获取群聊中的所有用户的会员信息,如果群聊界面中也包含有自己的好友,那么我们就会去判断,如果用户的会员信息在缓存中存在,则从缓存中获取,如果在缓存中不存在,则加入到一个请求集合中,批量查询会员信息后,合并本地缓存的会员信息和新的服务器的会员信息,将信息返回给群聊界面

1、创建会员实体类

private HashMap<Long, Vip> mVipCache = new HashMap<>();//作为缓存的类型

public static class Vip {
    //会员信息实体类
}

2、模拟从本地获取数据

  1. 遍历批量的uid参数,从缓存和网络中获取会员信息的数据
  2. 如果本地数据存在,则需要将缓存的会员信息加入到vipInfo
  3. 如果本地数据不存在,则需要将uid加入到请求列表mRequestList
  4. 最后合并本地缓存数据和网络请求数据
/**
 * 从本地缓存中获取数据
 */
public Observable<HashMap<Long, Vip>> getVipFromCache(List<Long> uids) {
    List<Long> mRequestList = new ArrayList<>();
    HashMap<Long, Vip> vipInfo = new HashMap<>();

    for (Long uid : uids) {
        if (mVipCache.containsKey(uid)) {
            Log.e("TAG", "从本地获取数据:" + uid + "用户");
            vipInfo.put(uid, mVipCache.get(uid));//如果缓存中有数据,则从本地中取出
        } else {
            Log.e("TAG", "从网络获取数据:" + uid + "用户");
            mRequestList.add(uid);//如果缓存中没有数据,则加入到网络请求列表中
        }
    }

    if (mRequestList.isEmpty()) {
        //如果请求列表中为空,则直接返回缓存的数据
        return Observable.just(vipInfo);
    }

    //合并缓存的数据和网络获取的数据
    return Observable.merge(Observable.just(vipInfo), getVipFromWeb(mRequestList));
}

3、模拟从服务器获取数据

  1. 睡眠2s钟,用于模拟网络请求的耗时时间
  2. 模拟后台返回的数据,并加入到缓存列表中
  3. 返回新的事件流
/**
 * 从网络上批量查询Vip信息
 */
public Observable<HashMap<Long, Vip>> getVipFromWeb(List<Long> uids) {
    //由于这里没有对应的接口,所以模拟请求网络数据
    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

    //模拟返回的数据
    HashMap<Long, Vip> vip = new HashMap<>();
    for (Long uid : uids) {
        //后台返回的数据进行赋值
        Vip vipInfo = new Vip();
        //vipInfo.xxx = WebValue;
        //vipInfo.xxx = WebValue;
        vip.put(uid, vipInfo);
    }

    //缓存到本地
    mVipCache.putAll(vip);

    return Observable.just(vip);
}

4、模拟合并本地数据和服务器数据

  1. 模拟应用场景,查询当前好友列表的会员信息
  2. 模拟应用场景,进入群聊页面,查询群聊列表的会员信息
  3. 模拟应用场景,定时更新会员的信息
/**
 * 模拟合并本地信息和服务器信息
 */
public void vip(View view) {
    List<Long> uids = new ArrayList<>();

    Log.e("TAG", "第一次查询,进入好友列表界面,查询好友列表的会员信息");
    uids.add(1L);
    uids.add(2L);
    getVipFromCache(uids);
    uids.clear();

    Log.e("TAG", "第二次查询,进入群聊界面,查询群聊中的会员信息");
    uids.add(1L);
    uids.add(3L);
    uids.add(4L);
    getVipFromCache(uids);
    uids.clear();

    Log.e("TAG", "第三次查询,定时更新最新会员信息,更新所有缓存里的数据");
    uids.add(1L);
    uids.add(2L);
    uids.add(3L);
    uids.add(4L);
    mVipCache.clear();
    getVipFromCache(uids);
}

5、输出结果

达到我们预期的设想和最优的解决方案

第一次查询,进入好友列表界面,查询好友列表的会员信息
从网络获取数据:1用户
从网络获取数据:2用户
第二次查询,进入群聊界面,查询群聊中的会员信息
从本地获取数据:1用户
从网络获取数据:3用户
从网络获取数据:4用户
第三次查询,定时更新最新会员信息,更新所有缓存里的数据
从网络获取数据:1用户
从网络获取数据:2用户
从网络获取数据:3用户
从网络获取数据:4用户
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,089评论 1 32
  • 第一部分 HTML&CSS整理答案 1. 什么是HTML5? 答:HTML5是最新的HTML标准。 注意:讲述HT...
    kismetajun阅读 27,422评论 1 45
  • ”最近一段时间一直在追剧,因为我特别喜欢姚晨,她演的剧很霸气,人物个性鲜明,真实率真不做作,对角色的把握非常到位,...
    芊浅好开心阅读 205评论 0 5
  • 皇太极还没来得及入主中原呢,突然的,中风,死了。皇太极死的突然,还没来得及安排接班人,就撒手人寰了。结果皇太极的弟...
    梅香书宛阅读 382评论 0 0
  • 也不知从什么时候开始,朋友圈里的结婚证书和婚纱照越来越多了,曾经一起玩的没心没肺的小伙伴们,穿上西装革履或者一席白...
    锡平君阅读 402评论 0 5