asynctaskの使い方の詳細

6929 ワード

Androidアプリケーションの開発には、単一スレッドモデルの原則を遵守する必要があります.Android UI操作はスレッドが安全ではなく、UIスレッドで実行する必要があります.単一スレッドモデルでは、常に2つの法則を覚えておく必要があります:1.UIスレッド2をブロックしないでください.UIスレッドでのみAndroid UIツールパッケージにアクセスすることを確認します.プログラムが最初に起動すると、Androidは対応するメインスレッド(Main Thread)を同時に起動します.メインスレッドは主にUIに関連するイベントを処理します.例えば、ユーザーのボタンイベント、ユーザーが画面に接触したイベント、画面描画イベントなど、関連するイベントを対応するコンポーネントに配布して処理します.したがって、プライマリスレッドはUIスレッドとも呼ばれることが多い.例えば、インターネットからウェブページを取得する、1つのTextViewにそのソースコードを表示する、このようなネットワーク操作に関わるプログラムは一般的に1つのスレッドを開いてネットワークアクセスを完了する必要があるが、ページのソースコードを取得した後、直接ネットワーク操作スレッドでTextViewを呼び出すことはできない.setText()の.他のスレッドではプライマリUIスレッドメンバーに直接アクセスできないためです.Androidは、他のスレッドでUIスレッドにアクセスする方法をいくつか提供します.Activity.runOnUiThread( Runnable ) View.post( Runnable ) View.postDelayed(Runnable,long)Hanlderのようなクラスや方法は、コードを複雑にし、理解しにくくなります.しかし、複雑な操作を実現し、UIを頻繁に更新する必要がある場合は、さらに悪化します.この問題を解決するために、Android 1.5はツールクラスを提供しています.AsyncTaskは、ユーザーインタフェースとインタラクティブに長時間実行する必要があるタスクの作成をより簡単にします.相対的にAsyncTaskはより軽量で,スレッドやHandlerを介さずに簡単な非同期処理に適している.AsyncTaskは抽象クラスです.AsyncTaskは3つの汎用型Params,Progress,Resultを定義した.Paramsは、HTTPリクエストのURLなど、タスク実行の入力パラメータを起動します.Progressバックグラウンドタスクの実行率.Resultバックグラウンドでタスクを実行して最終的に戻った結果、たとえばString.AsyncTaskの実行は4つのステップに分けられ、各ステップはコールバックメソッドに対応しており、これらのメソッドはアプリケーションによって呼び出されるべきではなく、開発者が必要とするのはこれらのメソッドを実現することです.1)サブクラス化AsyncTask 2)は、実際のバックグラウンド操作を実行する前にUI threadによって呼び出されるAsyncTaskで定義される次の1つまたは複数の方法onPreExecute()を実装する.インタフェースに進捗バーを表示するなど、この方法で準備をすることができます.doInBackground(Params...), onPreExecuteメソッドが実行されるとすぐに実行され、バックグラウンドスレッドで実行されます.ここでは、主に時間のかかるバックグラウンド計算の実行を担当します.publishProgressメソッドを呼び出して、リアルタイムのタスクの進捗状況を更新できます.この方法は抽象的な方法であり,サブクラスは実現しなければならない.onProgressUpdate(Progress...),publishProgressメソッドが呼び出されると、UI threadはこのメソッドを呼び出して、タスクの進捗状況をインタフェースに表示します.たとえば、進捗バーで表示します.onPostExecute(Result)は、doInBackgroundの実行が完了すると、onPostExecuteメソッドがUI threadによって呼び出され、バックグラウンドの計算結果がこのメソッドによってUI threadに渡される.AsyncTaskクラスを正しく使用するには、以下のいくつかのガイドラインを遵守する必要があります.1)Taskのインスタンスは、UI threadに2)executeメソッドを作成する必要があります.UI threadで3を呼び出す必要があります.手動でonPreExecute()を呼び出さないでください.onPostExecute(Result)、doInBackground(Params...)を呼び出さないでください.onProgressUpdate(Progress...)これらの方法4)taskは1回しか実行できません.そうしないと、複数回呼び出されたときに異常doInBackgroundメソッドとonPostExecuteのパラメータが対応する必要があります.この2つのパラメータは、AsyncTaskが宣言した汎用パラメータのリストで指定されています.1つ目はdoInBackgroundが受け入れたパラメータで、2つ目は進捗を表示するパラメータで、3つ目はdoInBackgroundが返し、onPostExecuteが入力したパラメータです.インターネットからWebページを取得し、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 {
// , 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]);
}

}

}

最後に、AsyncTaskはスレッドを完全に置き換えることができず、いくつかの論理が複雑であるか、バックグラウンドで繰り返し実行する必要がある論理ではスレッドが必要になる可能性があることを説明する必要があります.