5

私のアプリの1つに、バックグラウンドタスクを実行する必要があるシナリオがあります。そのために、私は非同期タスクを使用しています。また、カスタム進行状況ダイアログを使用しています。以下は、カスタム進行状況ダイアログのレイアウトです

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/layout_root"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_gravity="center_vertical|center_horizontal"
    android:orientation="vertical" >

    <ProgressBar
        android:layout_width="60dp"
        android:layout_height="60dp"
        android:indeterminateDrawable="@drawable/progressloader" 
        android:layout_gravity="center"/>

    <TextView
        android:id="@+id/progressMessage"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:textColor="@color/black"
        android:textSize="18sp"
        android:text="Please wait...." />

</LinearLayout>

すべて正常に動作しますが、テキストをTextViewに設定しようとすると、javaNullPointerExceptionが発生します。

AsyncTaskコード

private class InitialSetup extends AsyncTask<String, Integer, Long> {

        ProgressDialog dialog = new ProgressDialog(getParent(),R.style.progressdialog);


        @Override
        protected void onPreExecute() {
            dialog.show();
            dialog.setContentView(R.layout.progressbar);

        }

        @Override
        protected Long doInBackground(String... urls) {
                    //    txtView.setText("Testing");    here I am getting the error
            fetchDetails();

            return 0;
        }

        @Override
        protected void onPostExecute(Long result) {

            if (this.dialog.isShowing()) {
                this.dialog.dismiss();
            }

            populateUI(getApplicationContext());
        }
    }

主な活動

public class SummaryActivity extends Activity {


final TextView txtView = (TextView)findbyid(R.id.progressMessage);
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.accountsummary);

              new InitialSetup().execute("");

    }
}
4

4 に答える 4

14

私が正しく理解している場合TextView、テキストを設定したいものはxmlファイルprogressbar.xml(つまりR.layout.progressbar)にあります。これTextViewは、( を使用して) コンテンツ ビューを設定すると取得できますsetContentView()。あなたのコードでは、この呼び出しが行われる前にそれを設定し、mussharapp のコードでは、彼はそれを早期に呼び出しています。つまり、 .setContentView(R.layout.accountsummary)を含まない呼び出しの後にそれを呼び出しますTextView。その結果、変数txtViewは NULL になり、NullPointerException.

あなたがすべきことは次のとおりです。

  • onPreExecuteが呼び出された後、変数 txtView を に設定しますsetContentView
  • Paresh Mayani の説明に基づく: runOnUiThread メソッドを使用します。

コードについては、以下をご覧ください。

private class InitialSetup extends AsyncTask<String, Integer, Long> {

        ProgressDialog dialog = new ProgressDialog(getParent(),R.style.progressdialog);
        // The variable is moved here, we only need it here while displaying the
        // progress dialog.
        TextView txtView;

        @Override
        protected void onPreExecute() {
            dialog.show();
            dialog.setContentView(R.layout.progressbar);
            // Set the variable txtView here, after setContentView on the dialog
            // has been called! use dialog.findViewById().
            txtView = dialog.findViewById(R.id.progressMessage); 
        }

        @Override
        protected Long doInBackground(String... urls) {
            // Already suggested by Paresh Mayani:
            // Use the runOnUiThread method.
            // See his explanation.
            runOnUiThread(new Runnable() {
               @Override
               public void run() {
                  txtView.setText("Testing");       
               }
            });

            fetchDetails();
            return 0;
        }

        @Override
        protected void onPostExecute(Long result) {

            if (this.dialog.isShowing()) {
                this.dialog.dismiss();
            }

            populateUI(getApplicationContext());
        }
    }
于 2012-04-17T18:16:02.087 に答える
12

はい、doInBackground() メソッド内で TextView を設定しようとしており、これは許可されていないため、

なぜ許可されないのですか?UI メイン スレッドである 1 つのスレッドしか実行されておらず、スレッド プロセスから UI を更新することは許可されていないためです。ここで詳細情報をお読みください:痛みのないスレッディング

したがって、doInBackground() メソッド内で TextView を設定したい場合は、runOnUiThreadメソッド内で UI 更新操作を行うという解決策があります。

それ以外の場合は、AsyncTask クラスの doInBackground() メソッドではなく、onPostExecute() メソッド内ですべての UI 表示/更新関連操作を実行することをお勧めします。

于 2012-04-17T18:02:57.550 に答える
1
(TextView)findViewByid(R.id.progressMessage);

コマンド setContentView() の後にのみ実行する必要があります。

TextView txtView;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.accountsummary);
    **txtView = (TextView)findbyid(R.id.progressMessage);**


    new InitialSetup().execute("");

}

また、メイン UI スレッドでのみ UI 要素を変更できます。doInBackground() はメイン UI スレッドにはありません。onPostExecute で UI を変更する

public class InitialSetup extends AsyncTask<String, Integer, Long> {

        private Activity activity;
        ProgressDialog progressDialog;

        public InitialSetup(Activity activity) {
            this.activity = activity;
        }




        @Override
        protected void onPreExecute() {
            progressDialog = new ProgressDialog(activity);
            progressDialog.setMessage("Starting task....");
            progressDialog.show();    
        }

        @Override
        protected Long doInBackground(String... urls) {
            // do something

            //        

            return 0;
        }

        @Override
        protected void onPostExecute(Long result) {
            progressDialog.dismiss();
             //Perform all UI changes here
            **textView.setText("Text#2");**
        }
    }
于 2012-04-17T16:28:50.620 に答える
1

説明は正しいです。UI を作成するスレッド以外のスレッドで UI を変更してはなりません。しかし、AsyncTask には、というメソッドがあります。

onProgressUpdate()

これは常に UI スレッドで実行されます。したがって、dennisg による変更に基づいて、コードは次のようになります。

private class InitialSetup extends AsyncTask<String, String, Long> {

    ProgressDialog dialog = new ProgressDialog(getParent(),R.style.progressdialog);
    // The variable is moved here, we only need it here while displaying the
    // progress dialog.
    TextView txtView;

    @Override
    protected void onPreExecute() {
        dialog.show();
        dialog.setContentView(R.layout.progressbar);
        // Set the variable txtView here, after setContentView on the dialog
        // has been called! use dialog.findViewById().
        txtView = dialog.findViewById(R.id.progressMessage); 
    }

    @Override
    protected Long doInBackground(String... urls) {
        publishProgress("Testing");

        fetchDetails();

        return 0;
    }

    @Override
    protected void onPostExecute(Long result) {

        if (this.dialog.isShowing()) {
            this.dialog.dismiss();
        }

        populateUI(getApplicationContext());
    }

    @Override
    protected void onProgressUpdate(String... update) {
        if (update.length > 0)
            txtView.setText(update[0]); 
    }
}

onProgressUpdate のパラメーターの型は、AsyncTask で指定された 2 番目の型であることに注意してください。

追加:コードをより堅牢にするために、テキストを設定する前に進行状況ダイアログがまだ存在するかどうかを確認する必要があります。

于 2013-05-30T11:07:52.543 に答える