你知道RxJava也可以实现AsyncTask吗?

使用RxJava实现异步操作(AsyncTask)

常见的异步操作我们可以联想到AsyncTask或者handler,其实google创造出的目的也就是为了让代码更加清晰明了,让代码更加简洁.
而Rx系列的出现也就为了实现代码的逻辑清晰,结构简单问题.在gitHub上的介绍是 a library for composing asynchronous and event-based programs using observable sequences for the Java VM"(一个在 Java VM 上使用可观测的序列来组成异步的、基于事件的程序的库)没错,就是这样,我们可以通过他的操作符,通过他的线程切换,来实现各种Android上遇到的问题.如异步任务,事件总线,数据处理,Retrofit框架等.

提出需求

AsyncTask是异步操作最常见的处理方式,几乎只要涉及到网络请求的,或者耗时操作的,都会使用到AsyncTask,下面看看一个小需求.

  1. 给一个接口(查询IP)
    http://ip.taobao.com/service/getIpInfo.php
  2. 请求参数是
    IP地址: ip=192.168.0.1
  3. 功能实现
    请求时弹出进度框,请求完成隐藏进度框,并Toast出获取到的数据(需要将数据封装成bean).

常见的异步操作AsyncTask

下面我们使用AsyncTask来实现提出的需求.talk is cheap,请看代码.

//点击发出请求
comTitleStartTv.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                asyncTask();
            }
        });

具体的AsyncTask处理如下.

private void asyncTask() {
        new AsyncTask<String, Void, IPInfo>() {
            @Override
            protected IPInfo doInBackground(String... params) {
                // 在后台请求接口,并对请求到的json进行校验,解析成json
                String resultJsonStr = OkHttpUtil.get(Constants.ipUrl + getParam(params));
                IPInfo infos = null;
                if (resultJsonStr != null && !"".equals(resultJsonStr)) {
                    infos = new Gson().fromJson(resultJsonStr, IPInfo.class);
                }
                return infos;
            }

            @Override
            protected void onPreExecute() {
                super.onPreExecute();
          // 请求过程耗时操作,弹出Loading的进度框.
                if (dialog == null) {
                    dialog = ProgressDialog.show(MainActivity.this, "Loading...", "正在加载...", true, false);
                }
            }

            @Override
            protected void onPostExecute(IPInfo info) {
       // 请求完成,关闭进度框,并Toask请求后的结果数据.
                if (dialog != null)
                    CHelper.disimissDialog(dialog);

                if (info != null) {
                    Toast.makeText(MainActivity.this, "AsyncTask:" + info.toString(), Toast.LENGTH_SHORT).show();
                }
            }
        }.execute("192.168.0.1");
    }

    protected String getParam(String... p) {
        return "?ip=" + p[0];
    }

下面是AsyncTask的效果.
这里写图片描述

RxJava封装的异步操作

下面使用到了RxJava的操作符有线程切换subscribeOn,observeOn,doOnSubscribe,doOnCompleted,doOnError.

  • 封装BaseRxTask,主要是抽取公共方法和封装共性的功能.
    如: getBaseDilogView 主要是封装进度框,处理了io和UI线程.
/**
 * @Author: Relice
 * BaseRX 异步任务基类
 */
public abstract class BaseRxTask<D, P> {

    /**
     * 参数数组
     */
    protected P[] p;
    protected Context mContext;
    private boolean needDialog = false;
    private ProgressDialog dialog;

    /**
     * 接口不需要传参数 复写此方法
     */
    protected BaseRxTask(Context context, boolean needDialog) {
        this.mContext = context;
        this.needDialog = needDialog;
    }

    /**
     * 接口需要传参数调 复写此方法
     *
     * @param context 上下文
     * @param needDialog 是否显示进度
     * @param p 参数数组
     */
    @SafeVarargs
    protected BaseRxTask(Context context, boolean needDialog, P... p) {
        this.mContext = context;
        this.needDialog = needDialog;
        this.p = p;
    }

    protected Observable<D> doInBackgroundObserVable() {
        return doInBackground();
    }

    public Observable<D> execute() {
        return getBaseDilogView();
    }

    /**
     * 进度View
     *
     * @return
     */
    private Observable<D> getBaseDilogView() {
        Observable<D> tObservable = doInBackgroundObserVable();
 //subscribeOn(Schedulers.io())事件所产生的线程,也就是subscribe所发生的线程.
        return tObservable.subscribeOn(Schedulers.io())
 //doOnSubscribe的作用是如它后面有subscribeOn(),那么它将执行
//离它最近的subscribeOn()所指定的线程,也就是下面的              //observeOn(AndroidSchedulers.mainThread())线程.
                .doOnSubscribe(new Action0() {
                    @Override
                    public void call() {
                        if (needDialog && mContext != null) {
                            dialog = ProgressDialog.show(mContext, "Loading...", "正在加载...", true, false);
                        }
                    }
// 注意这里添加了doOnCompleted,doOnError两个预处理的操作,
//所以在后面对Observable的订阅就要用subscribe(new Observer),如果使用subscribe(new Action)会报错.
                }).doOnCompleted(new Action0() {
                    @Override
                    public void call() {
                        CHelper.disimissDialog(dialog);
                    }
                }).doOnError(new Action1<Throwable>() {
                    @Override
                    public void call(Throwable throwable) {
                        CHelper.disimissDialog(dialog);
                    }
                })
//为了让进度框可以在UI线程里执行,影响了上面doOnSubscribe的线程
                .observeOn(AndroidSchedulers.mainThread());
    }

    protected abstract Observable<D> doInBackground();

    /**
     * 接口没参数空实现即可
     *
     * @param p 参数数组
     */
    protected abstract String getParam(P... p);

    /**
     * @return BaseUrl
     */
    protected abstract String getBaseUrl();
}
  • 实现BaseRxTask
    因为在基类对所以的操作都封装好了,所以对BaseRxTask的实现就很简单了,代码中都有注释.

    public class GetRxIPInfoTask extends BaseRxTask<IPInfo, String> {
    protected GetRxIPInfoTask(Context context, boolean needDialog) {
        super(context, needDialog);
    }
    
    public GetRxIPInfoTask(Context context, boolean needDialog, String... p) {
        super(context, needDialog, p);
    }
    
    @Override
    public Observable<IPInfo> execute() {
        return super.execute();
    }
    
    @Override
    protected Observable<IPInfo> doInBackground() {
        return Observable.create(new Observable.OnSubscribe<IPInfo>() {
            @Override
            public void call(Subscriber<? super IPInfo> subscriber) {
                if (!subscriber.isUnsubscribed()) {
                    String resultJsonStr = OkHttpUtil.get(nullToEmpty(getBaseUrl()) + nullToEmpty(getParam(p)));
                    if (resultJsonStr != null && !"".equals(resultJsonStr)) {
                        IPInfo infos = new Gson().fromJson(resultJsonStr, IPInfo.class);
                        subscriber.onNext(infos);
                        subscriber.onCompleted();
                    } else {
                        subscriber.onCompleted();
                    }
                }
            }
        });
    }
    
    private String nullToEmpty(String str) {
        return (str == null) ? "" : str;
    }
    
    /**
     * ip
     *
     * @param p 参数数组
     */
    @Override
    protected String getParam(String... p) {
        return "?ip=" + p[0];
    }
    
    /**
     * @return base url
     */
    @Override
    protected String getBaseUrl() {
        return Constants.ipUrl;
    }
    }
  • 如何去调用
    机智的你们可以看到在基类中的execute()方法封装了 进度框的功能,

 public Observable<D> execute() {
        return getBaseDilogView();
    }

具体调用看下面代码

 private void rxAsyncTask() {
        new GetRxIPInfoTask(MainActivity.this, true, new String[]{"192.168.0.1"})
                .execute()
                //提前过滤掉不想要的数据
//                .filter(new Func1<IPInfo, Boolean>() {
//                    @Override
//                    public Boolean call(IPInfo ipInfo) {
//                        return ipInfo == null;
//                    }
//                })
                .subscribe(new Observer<IPInfo>() {
                    @Override
                    public void onCompleted() {
                        System.out.println("onCompleted");
                    }

                    @Override
                    public void onError(Throwable e) {
                        System.out.println("Throwable:" + e.getMessage());
                    }

                    @Override
                    public void onNext(IPInfo infos) {
                        System.out.println("onNext:" + infos.toString());
                        Toast.makeText(MainActivity.this, "RxAsyncTask:" + infos.toString(), Toast.LENGTH_SHORT).show();
                    }
                });
    }

最后我们看看RxAsyncTask的效果.
这里写图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值