AsyncTask продолжает сбой демо-приложения, работающего на эмуляторе Android

В настоящее время я изучаю разработку приложений для Android, работая над книгой, и в настоящее время я борюсь с чем-то, что на первый взгляд кажется довольно простым. Интерфейс состоит из кнопки и текстового представления. Идея упражнения состоит в том, чтобы продемонстрировать, насколько полезно использование отдельного потока при выполнении трудоемких задач.

Я пробовал отладку, и массив values метода onProgressUpdate действительно содержит переданное ему значение, и сбой всегда происходит при попытке установить текст для myTextView. Буду благодарен за любую помощь.

Код:

package com.ebookfrenzy.asyncdemo;

import androidx.appcompat.app.AppCompatActivity;
import android.os.AsyncTask;

import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {
    private TextView myTextView;
    private String tag = "Test";

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

    public void buttonClick(View view) {
        AsyncTask task = new MyTask().execute();
    }

    private class MyTask extends AsyncTask<String,Integer,String> {
        @Override
        protected void onPreExecute() {

        }

        @Override
        protected void onProgressUpdate(Integer... values) {
            Log.i(tag,"publish: " + Integer.toString(values[0]));
            myTextView.setText("Counter = " + values[0]);
        }

        @Override
        protected void onPostExecute(String result) {
            myTextView.setText(result);
        }

        @Override
        protected String doInBackground(String... strings) {
            int i = 0;
            while (i <= 20) {
                Log.i(tag,"before publish: " + Integer.toString(i));
                publishProgress(i);
                try {
                    Thread.sleep(1000);
                    i++;
                } catch(Exception e) {

                }
            }

            return "Button Pressed";
        }
    }
}

Вывод Logcat (созданный с помощью элементов Log.i, добавленных мной для устранения неполадок) обычно приведен ниже, но если всплывает предупреждающее сообщение, сообщающее, что приложение продолжает сбой, элемент «перед публикацией» продолжает заполняться с увеличением i каждый раз, пока я не закрою предупреждающее сообщение:

2020-12-13 23:17:04.499 31729-31772/com.ebookfrenzy.asyncdemo I/Test: before publish: 0
2020-12-13 23:17:04.503 31729-31729/com.ebookfrenzy.asyncdemo I/Test: publish: 0

Раздел ошибок Logcat:

    --------- beginning of crash
2020-12-13 23:19:47.635 353-353/com.ebookfrenzy.asyncdemo E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.ebookfrenzy.asyncdemo, PID: 353
    java.lang.NullPointerException: Attempt to invoke virtual method 'void android.widget.TextView.setText(java.lang.CharSequence)' on a null object reference
        at com.ebookfrenzy.asyncdemo.MainActivity$MyTask.onProgressUpdate(MainActivity.java:34)
        at com.ebookfrenzy.asyncdemo.MainActivity$MyTask.onProgressUpdate(MainActivity.java:25)
        at android.os.AsyncTask$InternalHandler.handleMessage(AsyncTask.java:715)
        at android.os.Handler.dispatchMessage(Handler.java:106)
        at android.os.Looper.loop(Looper.java:193)
        at android.app.ActivityThread.main(ActivityThread.java:6669)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:858)
2020-12-13 23:19:47.639 353-353/com.ebookfrenzy.asyncdemo I/Process: Sending signal. PID: 353 SIG: 9
myTextView — это null, потому что вы никогда не присваиваете значение полю. Обратите внимание, что AsyncTask устарела, и уже несколько лет считается некачественным решением. Возможно, вы захотите найти более новую книгу.
CommonsWare 14.12.2020 00:33

Спасибо, это помогло. Я пропустил эту строку в примере кода в книге. Книга Android Studio 4.0 Development Essentials, которая была опубликована в июне 2020 года, но, возможно, это слепое пятно. Или, возможно, позже в главе упоминается, что класс устарел. Что следует использовать вместо этого?

maca2kx 14.12.2020 01:35

Интересный. Эта книга описывает привязку представлений в главе 18, если я правильно читаю оглавление, поэтому я удивлен, что этот пример опирается на findViewById(), поскольку эта книга не охватывает AsyncTask до главы 65. Я бы рекомендовал сопрограммы вместо AsyncTask (глава 66), но вы не программируете на Kotlin, хотя эта книга, кажется, описывает это. ¯\_(ツ)_/¯

CommonsWare 14.12.2020 01:42

Привязка представления была добавлена ​​после первоначальной публикации, поэтому автор сообщает, что другие примеры не обновлялись, но рекомендует попробовать. Я только что закончил главу AsyncTask, прежде чем выйти из системы, поэтому я не знаю, что будет дальше, но есть отдельные книги, доступные для java и kotlin, так что, может быть, вы просматривали содержимое версии kotlin?

maca2kx 14.12.2020 02:25

«Может быть, вы смотрели содержимое версии kotlin?» -- в нем была целая куча глав о Котлине, так что, по-видимому, так оно и есть. Я не заметил, что у них были разные издания для каждого языка. И я думал, что у меня иногда запутанная линейка книг... :-)

CommonsWare 14.12.2020 13:03
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
В компьютерном программировании биты играют важнейшую роль в представлении и манипулировании данными на двоичном уровне. Побитовые операции...
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Приходилось ли вам сталкиваться с требованиями, в которых вас могли попросить поднять тревогу или выдать ошибку, когда метод Java занимает больше...
Полный курс Java для разработчиков веб-сайтов и приложений
Полный курс Java для разработчиков веб-сайтов и приложений
Получите сертификат Java Web и Application Developer, используя наш курс.
0
5
178
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Я думаю, что myTextView должен быть нулевым.

Попробуй это.

if (myTextView != null){
  myTextView.setText("Counter = " + values[0]);
}

И вы должны инициализировать метод myTextView at onCreate().

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    myTextView = (TextView) findViewById(R.id.mytextviewid); // put 
}

Спасибо, проблема была в том, что я не инициализировал переменную myTextView.

maca2kx 14.12.2020 15:10

Другие вопросы по теме