RxAndroid、RxJava的PublishSubject改造Observable

简介: RxAndroid、RxJava的PublishSubject改造Observable在附录1的基础上,用RxAndroid、RxJava的PublishSubject改造Observable,同时用OkHttp实现网络请求。


RxAndroid、RxJava的PublishSubject改造Observable

在附录1的基础上,用RxAndroid、RxJava的PublishSubject改造Observable,同时用OkHttp实现网络请求。
改造集中在Java上层代码,如:

package zhangphil.app;

import android.app.Activity;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Bundle;
import android.support.annotation.LayoutRes;
import android.support.annotation.NonNull;
import android.support.annotation.Nullable;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.ListView;

import java.util.ArrayList;
import java.util.UUID;
import java.util.concurrent.Callable;

import io.reactivex.android.schedulers.AndroidSchedulers;
import io.reactivex.observers.DisposableObserver;
import io.reactivex.schedulers.Schedulers;
import io.reactivex.subjects.PublishSubject;
import okhttp3.OkHttpClient;
import okhttp3.Request;
import okhttp3.Response;

/**
 * 这个例子假设有一个应用场景:每点击一次按钮,然后就启动一个线程加载一个网络图片到ListView
 *
 * 本例的数据流动:button启动addItem,然后在addItem里面的fromCallable加载回来一个Bitmap。
 * 此时fromCallable返回的Bitmap经由RxAndroid机制自动链式丢给DisposableObserver的onNext
 */

public class MainActivity extends Activity {

    private final String TAG = String.valueOf(UUID.randomUUID());

    private ArrayList items = new ArrayList<>();
    private ItemAdapter mAdapter;

    private PublishSubject mPublishSubject;
    private OkHttpClient mOkHttpClient;

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mOkHttpClient = new OkHttpClient();
        mPublishSubject = PublishSubject.create();

        ListView listView = (ListView) findViewById(R.id.listView);
        mAdapter = new ItemAdapter(this, R.layout.item);
        listView.setAdapter(mAdapter);

        final String url = "http://avatar.csdn.net/9/7/A/1_zhangphil.jpg";
        Button button = (Button) findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //点击按钮增加一个ListView的item
                addItem(url);
            }
        });
    }

    //假设这里每次都是新启一个耗时操作
    private void addItem(final String url) {
        mPublishSubject.fromCallable(new Callable<Bitmap>() {
            @Override
            public Bitmap call() throws Exception {
                Bitmap bmp = null;

                //同步方法返回观察者需要的数据结果
                //在这里处理线程化的操作

                Request request = new Request.Builder().url(url).build();
                Response response = mOkHttpClient.newCall(request).execute();
                try {
                    if (response.isSuccessful()) {
                        byte[] bytes = response.body().bytes();
                        bmp = BitmapFactory.decodeByteArray(bytes, 0, bytes.length);
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }

                return bmp;
            }
        }).subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new DisposableObserver<Bitmap>() {

                    @Override
                    public void onNext(@NonNull Bitmap bmp) {
                        items.add(bmp);
                    }

                    @Override
                    public void onComplete() {
                        mAdapter.notifyDataSetChanged();
                    }

                    @Override
                    public void onError(Throwable e) {
                        Log.e(TAG, e.toString(), e);
                    }
                });
    }


    /**
     * 以下是常规的Android ListView数据添加和更新Adapter
     */

    private class ItemAdapter extends ArrayAdapter {
        private LayoutInflater inflater;
        private int resId;

        public ItemAdapter(@NonNull Context context, @LayoutRes int resource) {
            super(context, resource);
            inflater = LayoutInflater.from(context);
            resId = resource;
        }

        @NonNull
        @Override
        public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
            if (convertView == null) {
                convertView = inflater.inflate(resId, null);
            }

            ImageView imageView = (ImageView) convertView.findViewById(R.id.image);
            imageView.setImageBitmap(getItem(position));

            return convertView;
        }

        @Nullable
        @Override
        public Bitmap getItem(int position) {
            Bitmap bmp = (Bitmap) items.get(position);
            return bmp;
        }

        @Override
        public int getCount() {
            return items.size();
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();

        //取消所有Okhttp的网络请求
        mOkHttpClient.dispatcher().cancelAll();
    }
}


附录:
1,《RxAndroid、RxJava的fromCallable更新数据加载到ListView简例》链接地址:http://blog.csdn.net/zhangphil/article/details/66970893
2,OkHttp:https://github.com/square/okhttp

相关文章
|
Android开发
【RecyclerView】 十四、GridLayoutManager 网格布局管理器 ( GridLayoutManager.SpanSizeLookup 指定 item 元素占用网格个数 )
【RecyclerView】 十四、GridLayoutManager 网格布局管理器 ( GridLayoutManager.SpanSizeLookup 指定 item 元素占用网格个数 )
1581 0
【RecyclerView】 十四、GridLayoutManager 网格布局管理器 ( GridLayoutManager.SpanSizeLookup 指定 item 元素占用网格个数 )
|
存储 API Android开发
kotlin开发安卓app,使用webivew 触发 onShowFileChooser, 但只能触发一次,第二次无法触发,是怎么回事。 如何解决
在Android WebView开发中,`onShowFileChooser`方法用于开启文件选择。当用户只能选择一次文件可能是因为未正确处理选择回调。解决此问题需确保:1) 实现`WebChromeClient`并覆写`onShowFileChooser`;2) 用户选择文件后调用`ValueCallback.onReceiveValue`传递URI;3) 传递结果后将`ValueCallback`设为`null`以允许再次选择。下面是一个Kotlin示例,展示如何处理文件选择和结果回调。别忘了在Android 6.0+动态请求存储权限,以及在Android 10+处理分区存储。
|
JSON 小程序 JavaScript
微信小程序页面事件,下拉刷新事件和上拉触底事件
这篇文章介绍了微信小程序中如何实现下拉刷新和上拉触底事件,包括开启下拉刷新、配置下拉刷新样式、监听下拉刷新事件,以及监听上拉触底事件和配置上拉触底的距离。
|
小程序
微信小程序无法触发onReachBottom的解决方案
这篇文章提供了解决微信小程序中`onReachBottom`事件无法触发的问题的方案。问题的原因可能是`onReachBottom`方法重复,解决方案是删除系统自动生成的该方法。
|
监控 物联网 Java
打造高可用系统:深入了解心跳检测机制
本文介绍了分布式系统中**心跳检测**的重要机制,用于监测系统节点的健康状态和通信畅通。心跳检测通过定期发送信号,若节点在预定期限内未响应则视为可能失效。处理机制包括重试、报警和自动修复。文章还提到了**周期检测**和**累计失效检测**两种策略,并给出Java代码示例展示心跳检测实现。此外,列举了心跳检测在分布式数据库、微服务和物联网等场景的应用,以及优化策略如动态调整心跳频率和优化超时机制。最后,强调了心跳检测对系统稳定性和高可用性的关键作用。
1914 2
|
JavaScript Java Android开发
android studio中文乱码各种情况的解决办法
android studio中文乱码各种情况的解决办法
706 0
android studio中文乱码各种情况的解决办法
|
消息中间件 SQL Rust
为什么选择 Kotlin 重写后端服务?
为什么选择 Kotlin 重写后端服务?
961 0
为什么选择 Kotlin 重写后端服务?
|
算法 程序员
【算法训练-链表 七】【链表排序】:链表排序、链表的奇偶重排、重排链表
【算法训练-链表 七】【链表排序】:链表排序、链表的奇偶重排、重排链表
221 1
|
小程序
微信小程序onReachBottom事件使用
微信小程序onReachBottom事件使用
821 0
|
数据安全/隐私保护 Android开发