RxJava系列之线程切换实战

系列文章

RxJava系列之简介和观察者设计模式
RxJava系列之上游与下游
RxJava系列之常用创建型操作符
RxJava系列之常用变换操作符
RxJava系列之常用过滤操作符
RxJava系列之常用条件操作符
RxJava系列之常用合并操作符
RxJava系列之常用异常操作符
RxJava系列之线程切换实战
RxJava系列之背压模式
RxJava系列之配合Retrofit
RxJava系列之泛型高级
RxJava系列之手写create操作符
RxJava系列之手写create操作符增加泛型限定
RxJava系列之手写just操作符
RxJava系列之手写map操作符
RxJava系列之手写切换线程

线程切换

前面的文章都是写的demo代码,本篇文章可以实战使用RxJava的方式加载网络图片。RxJava可以方便的设置上游和下游的执行线程,从而达到异步处理的作用。

如何切换线程

RxJava默认执行在主线程。如果需要切换上游执行线程,使用subscribeOn操作符。如果需要切换下游执行线程,使用observeOn操作符。

 /**
     * todo 异步线程区域
     * Schedulers.io() :代表io流操作,网络操作,文件流,耗时操作
     * Schedulers.newThread()    : 比较常规的,普普通通
     * Schedulers.computation()  : 代表CPU 大量计算 所需要的线程
     *
     * todo main线程 主线程
     * AndroidSchedulers.mainThread()  : 专门为Android main线程量身定做的
     *
     * todo 给上游分配多次,只会在第一次切换,后面的不切换了(忽略)
     * todo 给下游分配多次,每次都会去切换
     *
     * @param view
     */
    public void r01(View view) {
        // RxJava如果不配置,默认就是主线程main
        // 上游
        Observable.create(new ObservableOnSubscribe<String>() {
            @Override
            public void subscribe(ObservableEmitter<String> e) throws Exception {
                Log.d(TAG, "上游 subscribe: " + Thread.currentThread().getName());

                e.onNext("");
            }
        }).subscribeOn(Schedulers.io()) // todo 给上游配置异步线程    // 给上游分配多次,只会在第一次切换,后面的不切换了
                .subscribeOn(AndroidSchedulers.mainThread()) // 被忽略
                .subscribeOn(AndroidSchedulers.mainThread()) // 被忽略
                .subscribeOn(AndroidSchedulers.mainThread()) // 被忽略
                .subscribeOn(AndroidSchedulers.mainThread()) // 被忽略
                // result: io 异步线程

          .observeOn(AndroidSchedulers.mainThread()) // todo 给下游配置 安卓主线程    // 给下游分配多次,每次都会去切换
                .observeOn(AndroidSchedulers.mainThread()) // 切换一次线程
                .observeOn(AndroidSchedulers.mainThread()) // 切换一次线程
                .observeOn(AndroidSchedulers.mainThread()) // 切换一次线程
                .observeOn(Schedulers.io()) // 切换一次线程
                // result: io 异步线程

                .subscribe(new Consumer<String>() { // 下游简化版
            @Override
            public void accept(String s) throws Exception {
                Log.d(TAG, "下游 subscribe: " + Thread.currentThread().getName());
            }
        });
    }

是否指定异步线程的影响

如果未指定异步线程,则上游发一次,下游接收一次。上游发一次,下游接收一次。
如果指定了异步线程,则符合多线程的执行顺序,具有不确定性。

/**
     * todo 同步 和 异步 执行流程
     * @param view
     */
    public void r02(View view) {

        // TODO 默认情况下  上游和下游都是main线程的情况下
        // 上游
        /*Observable.create(new ObservableOnSubscribe<Integer>() {
            @Override
            public void subscribe(ObservableEmitter<Integer> e) throws Exception {
                Log.d(TAG, "subscribe: 上游发送了一次 1 ");
                e.onNext(1);

                Log.d(TAG, "subscribe: 上游发送了一次 2 ");
                e.onNext(2);

                Log.d(TAG, "subscribe: 上游发送了一次 3 ");
                e.onNext(3);
            }
        }).subscribe(new Consumer<Integer>() { // 下游简化版
            @Override
            public void accept(Integer integer) throws Exception {
                Log.d(TAG, "下游 accept: " + integer);
            }
        });*/

        /**
         * 默认情况下 同步的想象
         * 09-05 16:23:46.064 8088-8088/com.netease.rxjavastudy D/MainActivity8: subscribe: 上游发送了一次 1
         * 09-05 16:23:46.064 8088-8088/com.netease.rxjavastudy D/MainActivity8: 下游 accept: 1
         * 09-05 16:23:46.064 8088-8088/com.netease.rxjavastudy D/MainActivity8: subscribe: 上游发送了一次 2
         * 09-05 16:23:46.064 8088-8088/com.netease.rxjavastudy D/MainActivity8: 下游 accept: 2
         * 09-05 16:23:46.064 8088-8088/com.netease.rxjavastudy D/MainActivity8: subscribe: 上游发送了一次 3
         * 09-05 16:23:46.064 8088-8088/com.netease.rxjavastudy D/MainActivity8: 下游 accept: 3
         */






        // TODO 配置好异步线程
        Observable.create(new ObservableOnSubscribe<Integer>() {
            @Override
            public void subscribe(ObservableEmitter<Integer> e) throws Exception {
                Log.d(TAG, "subscribe: 上游发送了一次 1 ");
                e.onNext(1);

                Log.d(TAG, "subscribe: 上游发送了一次 2 ");
                e.onNext(2);

                Log.d(TAG, "subscribe: 上游发送了一次 3 ");
                e.onNext(3);
            }
        }).subscribeOn(Schedulers.io()) // 给上游分配 异步线程
          .observeOn(AndroidSchedulers.mainThread()) // 给下游分配 主线程

          .subscribe(new Consumer<Integer>() { // 下游简化版
            @Override
            public void accept(Integer integer) throws Exception {
                Log.d(TAG, "下游 accept: " + integer);
            }
        });
        /**
         * 配置了 异步后
         * 09-05 16:26:03.574 9547-9690/com.netease.rxjavastudy D/MainActivity8: subscribe: 上游发送了一次 1
         * 09-05 16:26:03.574 9547-9690/com.netease.rxjavastudy D/MainActivity8: subscribe: 上游发送了一次 2
         * 09-05 16:26:03.574 9547-9690/com.netease.rxjavastudy D/MainActivity8: subscribe: 上游发送了一次 3
         * 09-05 16:26:03.574 9547-9547/com.netease.rxjavastudy D/MainActivity8: 下游 accept: 1
         * 09-05 16:26:03.574 9547-9547/com.netease.rxjavastudy D/MainActivity8: 下游 accept: 2
         * 09-05 16:26:03.575 9547-9547/com.netease.rxjavastudy D/MainActivity8: 下游 accept: 3
         */
    }

常规方式加载网络图片

使用常规方式加载图片,代码四分五裂,没有串起来。增加或删减功能维护比较困难。

private final String PATH = "http://pic33.nipic.com/20131007/13639685_123501617185_2.jpg";

    /**
     * todo 不使用RxJava去下载图片
     * @param view
     */
    public void r03(View view) {

        progressDialog = new ProgressDialog(this);
        progressDialog.setMessage("正在下载中...");
        progressDialog.show();

        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(2000);

                    URL url = new URL(PATH);
                    URLConnection urlConnection = url.openConnection();
                    HttpURLConnection httpURLConnection = (HttpURLConnection) urlConnection;
                    httpURLConnection.setConnectTimeout(5000);
                    int responseCode = httpURLConnection.getResponseCode();
                    if (HttpURLConnection.HTTP_OK == responseCode) {
                       Bitmap bitmap = BitmapFactory.decodeStream(httpURLConnection.getInputStream());
                       Message message = handler.obtainMessage();
                       message.obj = bitmap;
                       handler.sendMessage(message);
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }

    private Handler handler = new Handler(new Handler.Callback() {
        @Override
        public boolean handleMessage(Message msg) {
            Bitmap bitmap = (Bitmap) msg.obj;
            imageView.setImageBitmap(bitmap);

            // 隐藏加载框
            if (progressDialog != null)
                progressDialog.dismiss();
            return false;
        }
    });

RxJava的方式加载网络图片

加载网络图片这件事被封装成一个事件流,增加与删减功能维护起来非常清楚。

   /**
     * todo 使用RxJava去下载图片
     * @param view
     */
    public void r04(View view) {

        // 起点

        // 上游 被观察者 Observable
        Observable.just(PATH)  // 内部发射

        // 加载网路图片,String Path  变换  Bitmap
        .map(new Function<String, Bitmap>() {
            @Override
            public Bitmap apply(String s) throws Exception {

                try {
                    Thread.sleep(2000);

                    URL url = new URL(PATH);
                    URLConnection urlConnection = url.openConnection();
                    HttpURLConnection httpURLConnection = (HttpURLConnection) urlConnection;
                    httpURLConnection.setConnectTimeout(5000);
                    int responseCode = httpURLConnection.getResponseCode();
                    if (HttpURLConnection.HTTP_OK == responseCode) {
                        Bitmap bitmap = BitmapFactory.decodeStream(httpURLConnection.getInputStream());
                        return bitmap;
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }

                return null;
            }
        })
.		// 给图片加水印
        .map(new Function<Bitmap, Bitmap>() {
            @Override
            public Bitmap apply(Bitmap bitmap) throws Exception {
                // 给图片加水印
                Paint paint = new Paint();
                paint.setColor(Color.RED);
                paint.setTextSize(30);
                Bitmap bitmapSuccess = drawTextToBitmap(bitmap, "同学们大家好", paint, 60, 60);
                return bitmapSuccess;
            }
        })

        // 比如:增加一个 日志纪录功能,只需要添加要给 变换操作符
        .map(new Function<Bitmap, Bitmap>() {
            @Override
            public Bitmap apply(Bitmap bitmap) throws Exception {
                Log.d(TAG, "apply: 下载的Bitmap 是这个样子的" + bitmap);
                return bitmap;
            }
        })

        .subscribeOn(Schedulers.io()) // todo  给上游分配 异步线程
        .observeOn(AndroidSchedulers.mainThread()) // todo  给下游分配 主线程

        .subscribe(new Observer<Bitmap>() { // 下游
            @Override
            public void onSubscribe(Disposable d) {
                progressDialog = new ProgressDialog(MainActivity8.this);
                progressDialog.setMessage("RxJava下载图片中..");
                progressDialog.show();
            }

            @Override
            public void onNext(Bitmap bitmap) {
                if (imageView != null)
                    imageView.setImageBitmap(bitmap);
            }

            @Override
            public void onError(Throwable e) { // 发生异常
               //  if (imageView != null)
                   //  imageView.setImageResource(R.mipmap.ic_launcher); // 下载错误的图片
            }

            @Override
            public void onComplete() { // 终点
                if (progressDialog != null)
                    progressDialog.dismiss();
            }
        });
    }


    //图片上绘制文字
    private Bitmap drawTextToBitmap(Bitmap bitmap, String text, Paint paint, int paddingLeft, int paddingTop) {
        Bitmap.Config bitmapConfig = bitmap.getConfig();

        paint.setDither(true); // 获取跟清晰的图像采样
        paint.setFilterBitmap(true);// 过滤一些
        if (bitmapConfig == null) {
            bitmapConfig = Bitmap.Config.ARGB_8888;
        }
        bitmap = bitmap.copy(bitmapConfig, true);
        Canvas canvas = new Canvas(bitmap);

        canvas.drawText(text, paddingLeft, paddingTop, paint);
        return bitmap;
    }

总结

1.异步线程区域
Schedulers.io() :代表io流操作,网络操作,文件流,耗时操作
Schedulers.newThread() : 比较常规的,普普通通
Schedulers.computation() : 代表CPU 大量计算 所需要的线程
2.AndroidSchedulers.mainThread() : 专门为Android main线程量身定做的
3.给上游分配多次,只会在第一次切换,后面的不切换了(忽略)
4.给下游分配多次,每次都会去切换
5.如果不配置异步线程,上游发一次,下游接收一次,上游发一次,下游接收一次,上游发一次,下游接收一次
6.配置好异步线程,就是异步的表现
7.传统下载图片的写法,容易四分五裂
8.RxJava下载图片,基于事件流编程,一条链子,起点和终点

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
RxJava 切换线程内存泄漏的原因可能有以下几点: 1. 订阅关系未正确取消:在使用RxJava进行线程切换时,如果没有正确取消订阅关系,就可能导致内存泄漏。因为订阅关系会持有被观察者和观察者之间的引用,如果未及时取消订阅,就会导致观察者无法被释放,从而造成内存泄漏。 2. 使用错误的调度器:在RxJava中,切换线程需要使用Scheduler来指定要切换到的线程。如果使用了错误的调度器,可能导致线程资源无法正确释放,从而造成内存泄漏。 3. 长时间运行的任务:如果在使用RxJava进行线程切换时,执行的任务是一个长时间运行的任务,可能会导致内存泄漏。因为长时间运行的任务可能会持有一些资源,如果没有及时释放这些资源,就会造成内存泄漏。 为避免RxJava切换线程内存泄漏,可以采取以下几个措施: 1. 及时取消订阅关系:在不需要继续观察的时候,及时取消订阅关系,可以使用CompositeDisposable来管理多个订阅关系,确保在不需要时能够正确取消订阅。 2. 使用正确的调度器:在进行线程切换时,确保使用正确的调度器,比如使用Schedulers.io()进行IO操作,使用AndroidSchedulers.mainThread()进行主线程操作。 3. 避免长时间运行的任务:如果任务可能会持续较长时间,可以考虑使用takeUntil等操作符来限制任务的执行时间。 通过以上措施,可以有效避免RxJava切换线程内存泄漏的问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值