Aync Task使用

6870 ワード

原文:http://chengbs.iteye.com/blog/1151838
Androidアプリケーションを開発するには、単一スレッドモデルの原則を遵守しなければならない.Android UI操作はスレッドの安全ではなく、UIスレッドで実行されなければならない.シングルスレッドモデルでは常に二つの法則を覚えます.  1.UIスレッドをブロックしない  2.UIスレッドのみでAndroid UIツールパッケージにアクセスすることを確認する.        プログラムが最初に起動されると、Androidは対応するメインスレッドを同時に起動し、主スレッドは主にUIに関連するイベントを処理する.したがって、主スレッドは通常、UIスレッドと呼ばれる.        例えば、インターネットからウェブページを取得すると、TextViewでソースコードが表示されます.このようなネットワーク操作に関するプログラムは、一般的にはスレッドを開いてネットワークアクセスを完了する必要がありますが、ページのソースを取得した後、直接にネットワーク操作スレッドにTextView.setText()を使うことはできません.他のスレッドでは直接メインUIスレッドのメンバーにアクセスできません.
androidは、他のスレッドでUIスレッドにアクセスするいくつかの方法を提供する.  Activity.run OnUiThread(Runnable)  View.post(Runnable)  View.postDelayed(Runnable、long)  Hanlder  これらの種類や方法はあなたのコードを複雑にして理解しにくいです.しかし、いくつかの複雑な操作を実現する必要があり、頻繁にUIを更新する必要がある場合、これはより悪くなります.       この問題を解決するために、Android 1.5はユーザーインターフェースと対話する必要がある長時間の実行を作成するタスクをより簡単にするツールクラスを提供しています.相対的にAyncTaskはもっと軽いサイズで、簡単な非同期処理に適しています.スレッドとHandlerを介して実現できます.  AsyncTaskは抽象類です.AsyncTaskは三つの汎型Paraams、Proggress、Resultを定義します.  Paramsは、HTTP要求のURLなどのジョブ実行の入力パラメータを起動する.  Progressバックグラウンドタスク実行のパーセンテージ.  Resultバックグラウンドでは、Steringなどのタスクが最終的に戻ってきます.       AsyncTaskの実行は4つのステップに分けられています.各ステップは1つのコールバック方法に対応しています.これらの方法はアプリケーションによって呼び出すべきではなく、開発者が必要とするのはこれらの方法を実現することです.  1)サブクラス化AyncTask  2)AyncTaskで定義された次の方法またはいくつかを実現する.       OnPreExecute()は、実際のバックグラウンド動作を実行する前にUI threadによって呼び出される.この方法では,界面にプログレスバーを示すようないくつかの準備作業が可能である.      doInBackgroundは、OnePreExecute方法の実行後すぐに実行され、この方法はバックグラウンドスレッドで実行される.ここでは主に時間がかかるバックグラウンドの計算作業を行います.publish Prograssメソッドを呼び出して、リアルタイムのタスクの進捗を更新することができます.この方法は抽象的な方法であり、サブクラスは必ず実現しなければならない.      OProgress Update(Progess...)は、Progressメソッドが呼び出された後、UI threadは、この方法を呼び出して、作業の進捗状況をインターフェースに表示します.例えば、プログレスバーを通じて展示します.      onPostExecute(Result)は、doInBackgroundの実行が完了すると、onPostExecute方法がUI threadによって呼び出され、バックグラウンドの計算結果がUI threadに渡されます.  AyncTaskクラスを正確に使用するために、以下のいくつかの遵守しなければならない基準があります.  1)Taskの例はUI threadで作成しなければならない.  2)execute方法はUI threadで呼び出す必要があります.  3)手動でオンプレクス()、onPostExecute(Resoult)、doInBackground(Params...)、onProgress Update(Progess...)を呼び出すのはやめましょう.  4)このtaskは一回しか実行できません.そうでないと、何回も起動すると異常が発生します.        doInBackground方法とonPostExecuteのパラメータは、AyncTask声明の汎型パラメータリストで指定されています.最初はdoInBackgroundで受け入れられたパラメータで、第二は進捗を示すパラメータで、第三はdoInBackgroundとonPostExecuteから入ってきたパラメータです.
インターネットからウェブページを取得し、TextViewでソースコードを表示します.
package test.list;
import java.io.ByteArrayOutputStream;
import java.io.InputStream;
import java.util.ArrayList;

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;

import android.app.Activity;
import android.app.ProgressDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.os.AsyncTask;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;

public class NetworkActivity extends Activity{
    private TextView message;
    private Button open;
    private EditText url;

    @Override
    public void onCreate(Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.network);
       message= (TextView) findViewById(R.id.message);
       url= (EditText) findViewById(R.id.url);
       open= (Button) findViewById(R.id.open);
       open.setOnClickListener(new View.OnClickListener() {
           public void onClick(View arg0) {
              connect();
           }
       });

    }

    private void connect() {
        PageTask task = new PageTask(this);
        task.execute(url.getText().toString());
    }


    class PageTask extends AsyncTask<String, Integer, String> {
        //         , AsyncTask.exucute()  
        ProgressDialog pdialog;
        public PageTask(Context context){
            pdialog = new ProgressDialog(context, 0);   
            pdialog.setButton("cancel", new DialogInterface.OnClickListener() {
             public void onClick(DialogInterface dialog, int i) {
              dialog.cancel();
             }
            });
            pdialog.setOnCancelListener(new DialogInterface.OnCancelListener() {
             public void onCancel(DialogInterface dialog) {
              finish();
             }
            });
            pdialog.setCancelable(true);
            pdialog.setMax(100);
            pdialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
            pdialog.show();


        }
        @Override
        protected String doInBackground(String... params) {

            try{

               HttpClient client = new DefaultHttpClient();
               // params[0]     url
               HttpGet get = new HttpGet(params[0]);
               HttpResponse response = client.execute(get);
               HttpEntity entity = response.getEntity();
               long length = entity.getContentLength();
               InputStream is = entity.getContent();
               String s = null;
               if(is != null) {
                   ByteArrayOutputStream baos = new ByteArrayOutputStream();

                   byte[] buf = new byte[128];

                   int ch = -1;

                   int count = 0;

                   while((ch = is.read(buf)) != -1) {

                      baos.write(buf, 0, ch);

                      count += ch;

                      if(length > 0) {
                          //          ,  publishProgress()    
                          publishProgress((int) ((count / (float) length) * 100));
                      }

                      //      100ms
                      Thread.sleep(100);
                   }
                   s = new String(baos.toByteArray());              }
               //     
               return s;
            } catch(Exception e) {
               e.printStackTrace();

            }

            return null;

        }

        @Override
        protected void onCancelled() {
            super.onCancelled();
        }

        @Override
        protected void onPostExecute(String result) {
            //   HTML     
            message.setText(result);
            pdialog.dismiss(); 
        }

        @Override
        protected void onPreExecute() {
            //     ,            ,      
            message.setText(R.string.task_started);
        }

        @Override
        protected void onProgressUpdate(Integer... values) {
            //     
              System.out.println(""+values[0]);
              message.setText(""+values[0]);
              pdialog.setProgress(values[0]);
        }

     }

}
 
最後に、AyncTaskがスレッドを完全に置き換えることができないことを説明する必要があり、いくつかの論理が複雑であるか、あるいはバックグラウンドで繰り返し実行する必要があるロジックは、スレッドを必要として実現されるかもしれない.