调度程序抛出致命异常 - 在Android中的rxJava2

时间:2017-09-04 22:03:20

标签: java android multithreading rx-java2 android-mvp

我将自己介绍给使用MVP的Android中的rxJava2。

对于初始步骤,我创建了一个Activity,一个presenter和一个Model对象(每个对象都有一个接口类)。

在我的活动中,我创建了演示者,并尝试加载数据,在视图准备就绪时通知视图:

protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.my_activity);

    MvpPresenter presenter = new Presenter(this,
            AndroidSchedulers.mainThread());

    presenter.loadData();

在演示者内部,在loadData方法中,只需回调视图

@Override
public void loadData() {

    compositeDisposable.add(dataRepository.getDataList().
            subscribeOn(Schedulers.io()).
            observeOn(mainScheduler).
            subscribeWith(new DisposableSingleObserver<List<Book>>() {
                @Override
                public void onSuccess(@NonNull List<Book> bookList) {
                    if (!bookList.isEmpty())
                        view.displayBooks(bookList);
                    else
                        view.displayEmpty();
                }

                @Override
                public void onError(@NonNull Throwable e) {
                    view.displayError("boooom");
                }
            }));
    }

方法getBooks()如下(我考虑使用API​​,但到目前为止,为了简单起见,我只是对数据进行了硬编码)

public Single<List<Shelf>> getShelfList() {
    return Single.fromCallable(new Callable<List<Book>>() {
        @Override
        public List<Book> call() throws Exception {
            Book b1 = new Book();
            Book b2 = new Book();
            Book b3 = new Book();

            List<Book> bookList = Arrays.asList(b1,b2,b3);
            return bookList;
        }
    });
}

方法view.displayBooks(bookList)只有一个Toast消息(再次,为了简单起见)

public void displayBooksInShelf(List<Shelf> shelfList) {
        Toast.makeText(this, shelfList.size(), Toast.LENGTH_LONG).show();
    }

看起来非常简单,它应该像Toast一样正确地返回三本书......但是在执行时,它会引发异常:

java.lang.IllegalStateException: Fatal Exception thrown on Scheduler.
 at io.reactivex.android.schedulers.HandlerScheduler$ScheduledRunnable.run(HandlerScheduler.java:111)
 at android.os.Handler.handleCallback(Handler.java:751)
 at android.os.Handler.dispatchMessage(Handler.java:95)
 at android.os.Looper.loop(Looper.java:154)
 at android.app.ActivityThread.main(ActivityThread.java:6077)
 at java.lang.reflect.Method.invoke(Native Method)
 at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:866)
 at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:756)
Caused by: android.content.res.Resources$NotFoundException: String resource ID #0x2
 at android.content.res.Resources.getText(Resources.java:331)
 at android.widget.Toast.makeText(Toast.java:287)

在单一测试(Mockito)中,一切似乎都很好,但它一起失败了。我的嫌疑人是有某种线程问题。有人可以帮我解决这个问题吗?

另外,知道我想要改装,我使用正确的方法吗?或者可能应该在Model / Repository类中管理线程?

希望我提供的信息足以说明问题所在

提前致谢!

1 个答案:

答案 0 :(得分:1)

注意stacktrace中的这一行:

Caused by: android.content.res.Resources$NotFoundException: String resource ID #0x2

您正在尝试使用无效的资源ID访问String资源。在这里发生的地方:

Toast.makeText(this, shelfList.size(), Toast.LENGTH_LONG).show();

第二个参数必须是String消息 int资源ID。您在此处传递int,因此编译器选择具有资源ID的重载(​​显然无效)。

如果您只想打印int,则必须传递String个代表:

Toast.makeText(this, Integer.toString(shelfList.size()), Toast.LENGTH_LONG).show();