rx android vs mosby 模式加载数据 recyclerview

rx android vs mosby pattern load data recyclerview

我正在开发 Android 应用程序。我附加的代码正在创建一个回收视图。我们做的第一件事是创建一个异步任务,它将在 SQLite 数据库上获取数据并将其加载到适配器->recylcerview 中。当后台任务运行时,会向用户显示一个进度对话框。

public class HomeActivity extends AppCompatActivity
{
    private RecyclerView recycler;
    private RecyclerViewAdapter adapter;
    private SwipeRefreshLayout swipeRefresh;
    private progressDialog progressDialog;

     // ... some code here

 @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        // ... some code here


    createRecyclerView();
    loadRecyclerView();


    // ... some code here

    }


    private void loadRecyclerView()
    {
        new LoadingBackgroundTask().execute();
    }



    private void createRecyclerView()
{

    Context context = getApplicationContext();

    recycler = (RecyclerView) findViewById(R.id.recycle_view_home);
    recycler.setHasFixedSize(true);

    RecyclerView.LayoutManager lManager = new LinearLayoutManager(context);
    recycler.setLayoutManager(lManager);

    adapter = new RecyclerViewAdapter();

    recycler.setAdapter(adapter);
    recycler.setItemAnimator(new DefaultItemAnimator());

}

private class LoadingBackgroundTask extends AsyncTask<Void, Void, List<items>> {

    @Override
    protected void onPreExecute() {
        super.onPreExecute();
        progressDialog = ProgressDialog.show(HomeActivity.this, getString(R.string.dialog_load_list),getString(R.string.dialog_please_wait), false, false);

    }

    @Override
    protected List doInBackground(Void... params) {

        List<items> lists;
        //Data Source Class ( SQLite)
        ListDS listDS = new ListDS(getApplicationContext());
        list = listDS.getList();

        return list;
    }

    @Override
    protected void onPostExecute(List result) {
        super.onPostExecute(result);

        //it inserts de list on recyclerview performing animation
        adapter.animate(result);

        progressDialog.dissmiss();
        swipeRefresh.setRefreshing(false);
        recycler.scrollToPosition(0);
    }

}

}

到目前为止,还不错。但是,您可能知道这段代码有一些众所周知的问题;例如,如果我在 asynctask 发挥它的魔力时旋转屏幕,它会使应用程序崩溃。

我尝试了另一种我在谷歌上看到的替代方法,rxandroid

(对不起,如果我打错了,我是靠记忆)

public class HomeActivity extends AppCompatActivity
{
 private Subscriber suscriptor;
private progressDialog progressDialog;

 //some code ....

  @Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    suscriptor = new Subscriber() {
        @Override
        public void onCompleted() {
            progressDialog.dismiss();
            Log.d("SUSCRIPTOR","ON COMPLETE");
        }

        @Override
        public void onError(Throwable e) {
            Log.d("SUSCRIPTOR","ON ERROR");
        }

        @Override
        public void onNext(Object o) {
            adapter.animate((List<items>)o);

        }
    };


    Observable.create(
            new Observable.OnSubscribe<List<items>>() {
                @Override
                public void call(Subscriber<? super List<items>> sub) {
                progressDialog = ProgressDialog.show(HomeActivity.this,  getString(R.string.dialog_load_list),getString(R.string.dialog_please_wait), false, false);
                    List<items> lists;
                    //Data Source Class ( SQLite)
                    ListDS listDS = new ListDS(getApplicationContext());
                    list = listDS.getList();

                    sub.onNext(list);
                    sub.onCompleted();

                }

                @Override
                protected void finalize() throws Throwable {
                    super.finalize();
                    Log.d("OBSERAVBLE","FINALIZED");
                }
            })
    .observeOn(AndroidSchedulers.mainThread())
    .subscribeOn(Schedulers.newThread())
           .cache()
    .subscribe(suscriptor);

    }


@Override
public void onDestroy()
{
    if(suscriptor!=null)
    {
        if(!suscriptor.isUnsubscribed())
        {
            suscriptor.unsubscribe();
        }
    }

    super.onDestroy();
}

}

现在当我旋转屏幕时应用程序不再崩溃了。然而,observable 一直在后台工作,直到它完成,但是当我取消订阅以避免崩溃时,我没有正确收到结果。此外,即使 observable 像我提到的那样继续工作,进度条也会消失。

寻找解决方案,我发现有一个名为 "Ted Mosby" 的模式似乎可以解决我的问题。虽然它看起来很有希望,但我认为对于一些我觉得不值得的东西来说编码太多了,rxandroid 可能有一个解决方法。

所以,我的问题是,我怎样才能得到我想要的东西,而不会陷入对我的目的来说太大的体系结构编码混乱中?如果你们解决了这个问题,你能举个例子吗?你认为我错了,我应该实施 TedMosby 模式吗?

您的 Observable 应该自己处理取消订阅的事实。有两种机制:

  • 检查 subscriber.isUnsubscribed。您可以在 "heavy" 步
  • 之间或之后进行
  • 增加退订回调。您可以使用它来停止长时间的 运行 操作、释放资源等

看看这段代码:

Observable.create(
        new Observable.OnSubscribe<List<items>>() {
            @Override
            public void call(Subscriber<? super List<items>> sub) {
                sub.add(Subscriptions.create(new Action0() {
                    @Override
                    public void call() {
                        cancelLongRunningOperationIfItStillRunning();
                    }
                }));
                if (!sub.isUnsubscribed()) {
                    //start long running operation here
               }
            }
        })
        .doOnSubscribe(new Action0() {
            @Override
            public void call() {
            }
        })

您不应该在 Observable 中引用您的 activity/context/progress 对话框等。如果您想产生一些副作用,请改用 doOnSubscribe

Mosby 是一个 Model-View-Presenter (MVP) 库.所以你命名为 "ted mosby pattern" 的模式实际上是 MVP。

但你还没有理解什么是 MVP。这与保留异步 运行 任务无关,甚至认为这可以通过 Mosby 实现。 MVP 是关于关注点分离的。 View只是显示UI个元素,Presenter控制着View,即presenter告诉view:现在显示进度对话框,现在隐藏进度对话框等。换句话说,presenter控制视图的状态. Model 可以是异步任务或 RxJava Observable。然后 Presenter 取回结果并告诉视图显示它。您的代码被解耦为 3 层模型(也称为业务逻辑)以及 Presenter 和 View。优点是您可以更改视图(即用进度条小部件替换进度对话框)而无需触及任何加载数据的代码(Presenter 和业务逻辑)。此外,通过 MVP,您的代码变得可测试。

所以你应该比较的是:我应该使用 AsyncTask 还是 RxJava 来加载数据。使用 Mosby,您可以在演示器中执行您的 http 请求。完成方向更改后,演示者不会被破坏(因此后台任务不会被取消)。

但是,MVP 并不是万能的解决方案。如果您必须确保正确执行单个 http 调用(即注册社区),您应该考虑使用 android 服务。

你可以做的就是在 activity 中所做的任何事情,比如你的 AsycTaskRecyclerView,把它放在一个片段中,然后 setRetainInstance(true)onCreateView() 片段的方法并将该片段加载到您的 activity.

setRetainInstance(true) 不会让您的片段实例在屏幕旋转时销毁。