Мое приложение падает, когда я нажимаю кнопку «Плавающее действие»: только исходный поток, создавший иерархию представлений, может касаться его представлений. - PullRequest
0 голосов
/ 13 февраля 2019

Я создаю приложение, используя Room Database.В своей DetailActivity я установил плавающую кнопку действия, чтобы пометить фильм как любимый.Когда я нажимаю на ФАБ, приложение вылетает.Я предоставляю метод, в котором у меня появляется ошибка.

public void click (){
        if (isClicked) {
            MoviesExecutor.getInstance().diskIO().execute(new Runnable() {
                @Override
                public void run() {
                    mDb.moviesDao().removeMovie(movies.getMovieId());
                    favoriteFab.setImageResource(R.drawable.heart_favorite_border_black);

                }
            });
            isClicked = false;

        } else {MoviesExecutor.getInstance().diskIO().execute(new Runnable() {
            @Override
            public void run() {
                mDb.moviesDao().addMovie(movies);
                favoriteFab.setImageResource(R.drawable.heart_favorite_black);
            }
        });
            Toast.makeText(this, "added", Toast.LENGTH_LONG).show();
            isClicked = true;

        }

LOGS:

02-13 15:30:55.294 21789-21880/? E/AndroidRuntime: FATAL EXCEPTION: pool-2-thread-1
    Process: com.example.android.popularmovies2, PID: 21789
    android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
        at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:6556)
        at android.view.ViewRootImpl.invalidateChildInParent(ViewRootImpl.java:942)
        at android.view.ViewGroup.invalidateChild(ViewGroup.java:5082)
        at android.view.View.invalidateInternal(View.java:12758)
        at android.view.View.invalidate(View.java:12722)
        at android.view.View.invalidate(View.java:12706)
        at android.widget.ImageView.setImageDrawable(ImageView.java:479)
        at android.support.design.widget.FloatingActionButton.setImageDrawable(FloatingActionButton.java:483)
        at android.support.v7.widget.AppCompatImageHelper.setImageResource(AppCompatImageHelper.java:90)
        at android.support.design.widget.FloatingActionButton.setImageResource(FloatingActionButton.java:478)
        at com.example.android.popularmovies2.DetailActivity$4.run(DetailActivity.java:152)
        at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1113)
        at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:588)
        at java.lang.Thread.run(Thread.java:818)

Ответы [ 2 ]

0 голосов
/ 13 февраля 2019

Сообщение об ошибке очень очевидно, операции с БД почти гарантированно завершатся успешно, так что вы можете просто переместить эту строку: favoriteFab.setImageResource(R.drawable.heart_favorite_border_black); из запуска.Таким образом, ваш код будет выглядеть примерно так:

public void click (){
    if (isClicked) {
        MoviesExecutor.getInstance().diskIO().execute(new Runnable() {
            @Override
            public void run() {
                mDb.moviesDao().removeMovie(movies.getMovieId());
            }
        });

        favoriteFab.setImageResource(R.drawable.heart_favorite_border_black);
        isClicked = false;

    } else {MoviesExecutor.getInstance().diskIO().execute(new Runnable() {
        @Override
        public void run() {
            mDb.moviesDao().addMovie(movies);
        }
    });
    favoriteFab.setImageResource(R.drawable.heart_favorite_black);
    Toast.makeText(this, "added", Toast.LENGTH_LONG).show();
    isClicked = true;
}
0 голосов
/ 13 февраля 2019

Вы можете выполнять операции с БД в потоке ввода-вывода, но не просматривать связанные материалы.

Ваша эта строка:

favoriteFab.setImageResource(R.drawable.heart_favorite_border_black);

создает проблему.

Вы можетеработать только с представлениями из MainThread (поток пользовательского интерфейса)

Если ваш код лежит в Activity, то вы можете получить метод runOnUIThread()

runOnUiThread (new Thread(new Runnable() {  
        public void run() {

         }
}));

Соберите его вместе,

public void click() {
    if (isClicked) {
        MoviesExecutor.getInstance().diskIO().execute(new Runnable() {
            @Override
            public void run() {
                mDb.moviesDao().removeMovie(movies.getMovieId());
                runOnUiThread(new Thread(new Runnable() {
                    public void run() {
                        favoriteFab.setImageResource(R.drawable.heart_favorite_border_black);
                    }
                }));
            }
        });
        isClicked = false;

    } else {
        MoviesExecutor.getInstance().diskIO().execute(new Runnable() {
            @Override
            public void run() {
                mDb.moviesDao().addMovie(movies);
                runOnUiThread(new Thread(new Runnable() {
                    public void run() {
                        favoriteFab.setImageResource(R.drawable.heart_favorite_black);
                    }
                }));
            }
        });
        Toast.makeText(this, "added", Toast.LENGTH_LONG).show();
        isClicked = true;

    }
}
...