可能性のある複製:
AsyncTaskのアクティビティの進行状況ダイアログの更新
私は最初のAndroidアプリを開発しています。バックグラウンドタスク(この場合はサーバー上のhttp呼び出し)が発生している間にProgressDialogを表示する必要があります。これについて少し勉強しました。また、この主題に関連する他のスレッドもすでにチェックしています。
http://developer.Android.com/reference/Android/os/AsyncTask.html
アクティビティUIの読み込みが完了するまでAndroidはProgressDialogを表示します
http://Android-developers.blogspot.com/2009/05/painless-threading.html
とりわけ。
私は少しのコードを書くようになったよりも:
1)私のアクティビティでは、変数をProgressDialog型として宣言します
public class LoginActivity extends Activity {
public static final String TAG = "LoginActivity";
protected ProgressDialog progressDialog;
...
2)また、必要に応じてAsyncTaskを拡張する内部クラスを作成しました。ここでは、doInBackGroundで、サーバー側で、サーバーへのPOST http要求を行う静的メソッドを呼び出しますサーバーの応答を20秒間ブロックして、進行状況ダイアログを検証しました。
class EfetuaLogin extends AsyncTask<Object, Void, String> {
private final static String TAG = "LoginActivity.EfetuaLogin";
@Override
protected void onPreExecute()
{
Log.d(TAG, "Executando onPreExecute de EfetuaLogin");
}
@SuppressWarnings("unchecked")
@Override
protected String doInBackground(Object... parametros) {
Log.d(TAG, "Executando doInBackground de EfetuaLogin");
Object[] params = parametros;
HttpClient httpClient = (HttpClient) params[0];
List<NameValuePair> listaParametros = (List<NameValuePair>) params[1];
String result = null;
try{
result = HttpProxy.httpPost(AnototudoMetadata.URL_AUTENTICACAO, httpClient, listaParametros);
}catch (IOException e) {
Log.e(TAG, "IOException, Sem conectividade com o servidor do Anototudo! " + e.getMessage());
e.printStackTrace();
return result;
}
return result;
}
@Override
protected void onPostExecute(String result)
{
progressDialog.dismiss();
}
}
3)ボタンが押されたら、ProgressDialogを作成し、作成したAsyncTaskを呼び出します:
OnClickListener loginListener = new OnClickListener() {
public void onClick(View v) {
//next line should start progress dialog in main thread ?????
progressDialog = ProgressDialog.show(LoginActivity.this, "Login in", "Wait a moment please", true, false);
//next couple of lines should do an ascyn call to server
EfetuaLogin efetuaLogin = new EfetuaLogin();
efetuaLogin.execute(params);
try {
//recover the server response and sets time out to be 25seconds
sResposta = efetuaLogin.get(25, TimeUnit.SECONDS);
これは、AsyncTaskがバックグラウンドでサーバーに照会している間に進行状況ダイアログを表示すると想定されていたと思いますが、サーバー応答が到着するまで、ほんの少しの間(1秒未満) )進行状況が表示され、次のアクティビティが呼び出されます。
私が述べたように、私はこのコードを再確認しましたが、どこで間違ったのか見つけることができません。助言がありますか?
前もって感謝します。
こんにちは、このスレッドの最初の答えでCharlie Sheen(???)が示唆したように、コードを少し変更しようとしましたが、今は次のようになっています(残念ながら、これまでのところ期待どおりに動作していません):
OnClickListener loginListener = new OnClickListener() {
public void onClick(View v) {
//async call????????
new EfetuaLogin().execute(params);
...
そして、AsyncTaskの応答に対処するためのすべての作業を行うよりも:
class EfetuaLogin extends AsyncTask<Object, Void, String> {
private final static String TAG = "LoginActivity.EfetuaLogin";
@Override
protected void onPreExecute()
{
super.onPreExecute();
Log.d(TAG, "Executando onPreExecute de EfetuaLogin");
//inicia diálogo de progresso, mostranto processamento com servidor.
progressDialog = ProgressDialog.show(LoginActivity.this, "Autenticando", "Contactando o servidor, por favor, aguarde alguns instantes.", true, false);
}
@SuppressWarnings("unchecked")
@Override
protected String doInBackground(Object... parametros) {
Log.d(TAG, "Executando doInBackground de EfetuaLogin");
Object[] params = parametros;
HttpClient httpClient = (HttpClient) params[0];
List<NameValuePair> listaParametros = (List<NameValuePair>) params[1];
String result = null;
try{
result = HttpProxy.httpPost(AnototudoMetadata.URL_AUTENTICACAO, httpClient, listaParametros);
}catch (IOException e) {
Log.e(TAG, "IOException, Sem conectividade com o servidor do Anototudo! " + e.getMessage());
e.printStackTrace();
return result;
}
return result;
}
@Override
protected void onPostExecute(String result)
{
super.onPostExecute(result);
if (result == null || result.equals("")) {
progressDialog.dismiss();
Alerta
.popupAlertaComBotaoOK(
"Dados incorretos",
"Os dados informados não foram encontrados no Sistema! Informe novamente ou cadastre-se antes pela internet.",
LoginActivity.this);
return;
}
Log.d(TAG, "Login passou persistindo info de login local no device");
ContentValues contentValues = new ContentValues();
contentValues.put(AnototudoMetadata.LOGIN_EMAIL, sLogin);
contentValues.put(AnototudoMetadata.LOGIN_SENHA, sSenha);
contentValues.put(AnototudoMetadata.LOGIN_SENHA_GERADA, result);
LoginDB loginDB = new LoginDB();
loginDB.addLogin(LoginActivity.this, contentValues);
Log.d(TAG, "Persistiu info de login no device, redirecionando para menu principal do Anototudo");
Log.d(TAG, "O retorno da chamada foi ==>> " + result);
// tudo ok chama menu principal
Log.d(TAG, "Device foi corretametne autenticado, chamando tela do menu principal do Anototudo.");
String actionName = "br.com.anototudo.intent.action.MainMenuView";
Intent intent = new Intent(actionName);
LoginActivity.this.startActivity(intent);
progressDialog.dismiss();
}
}
完全なOnClickListener:
OnClickListener loginListener = new OnClickListener() {
public void onClick(View v) {
Log.d(TAG, "Usuario logado, chamando menu principal");
TextView tLogin = (TextView) findViewById(R.id.loginText);
TextView tSenha = (TextView) findViewById(R.id.senhaText);
String sLogin = tLogin.getText().toString();
String sSenha = tSenha.getText().toString();
if (sLogin.equals("") | sSenha.equals("")) {
Alerta.popupAlertaComBotaoOK("Campos Obrigatórios",
"Os campos Login e Senha são obrigatórios para autenticação do Anototudo.", LoginActivity.this);
return;
} else {
Pattern regEx = Pattern.compile(".+@.+\\.[a-z]+");
Matcher matcher = regEx.matcher(sLogin);
if (!matcher.matches()) {
Alerta.popupAlertaComBotaoOK("Formato e-mail inválido", "O formato do campo e-mail está inválido",
LoginActivity.this);
return;
}
}
List<NameValuePair> listaParametros = new ArrayList<NameValuePair>();
listaParametros.add(new BasicNameValuePair("login", sLogin));
listaParametros.add(new BasicNameValuePair("senha", sSenha));
Log.d(TAG, "valores recuperados dos campos de login e senha: " + sLogin + " | " + sSenha);
// Reutiliza cliente HttpClient disponibilizado pela Aplicação.
AnototudoApp atapp = (AnototudoApp) LoginActivity.this.getApplication();
HttpClient httpClient = atapp.getHttpClient();
//prepara lista de parametros para fazer chamada asíncrona no servidor para autenticar.
Object[] params = new Object[2];
params[0] = httpClient;
params[1] = listaParametros;
//faz chamada assincrona
new EfetuaLogin().execute(params);
}
};
ProgressDialog
をonPreExecute
に配置します。以下のサンプルコード:
private ProgressDialog pdia;
@Override
protected void onPreExecute(){
super.onPreExecute();
pdia = new ProgressDialog(yourContext);
pdia.setMessage("Loading...");
pdia.show();
}
@Override
protected void onPostExecute(String result){
super.onPostExecute(result);
pdia.dismiss();
}
onClickListener
で、次の行を中に入れてください:
new EfetuaLogin().execute(null, null , null);
うまくいった最後の解決策は、OnClickListener
からdoInBackground
実装のAsyncTask
メソッドまでのすべてのコードを取得することです。コードは次のようになります。
OnClickListener
:
OnClickListener loginListener = new OnClickListener() {
public void onClick(View v) {
/* Translation note: Original text: "Executando OnClickListener" */
Log.d(TAG, "OnClickListener has been called");
/* Translation note: Original text: "faz chamada assincrona" */
// Make an asynchronous call
new EfetuaLogin().execute();
}
};
すべての作業はEfetuaLogin
AsyncTask
実装で行われます:
class EfetuaLogin extends AsyncTask<Object, Void, String> {
private final static String TAG = "LoginActivity.EfetuaLogin";
protected ProgressDialog progressDialog;
@Override
protected void onPreExecute()
{
super.onPreExecute();
Log.d(TAG, "Executando onPreExecute de EfetuaLogin");
//inicia diálogo de progresso, mostranto processamento com servidor.
progressDialog = ProgressDialog.show(LoginActivity.this, "Autenticando", "Contactando o servidor, por favor, aguarde alguns instantes.", true, false);
}
@SuppressWarnings("unchecked")
@Override
/* Translation note: Original text: "Object... parametros"
protected String doInBackground(Object... parameters) {
/* Translation note: Original text: "Executando doInBackground de EfetuaLogin" */
Log.d(TAG, "Executing doInBackground of EfetuaLogin");
TextView tLogin = (TextView) findViewById(R.id.loginText);
TextView tSenha = (TextView) findViewById(R.id.senhaText);
String sLogin = tLogin.getText().toString();
String sSenha = tSenha.getText().toString();
if (sLogin.equals("") | sSenha.equals("")) {
/*
Translation notes:
1) "Campos Obrigatórios" -> "Required fields"
2) "Os campos Login e Senha são obrigatórios para autenticação do Anototudo." -> "Login and Password fields are required for Anototudo authentication."
Alerta.popupAlertaComBotaoOK("Required fields", "Login and Password fields are required for Anototudo authentication.", LoginActivity.this);
progressDialog.dismiss();
return null;
} else {
Pattern regEx = Pattern.compile(".+@.+\\.[a-z]+");
Matcher matcher = regEx.matcher(sLogin);
if (!matcher.matches()) {
/*
Translation notes:
1) "Formato e-mail inválido" -> "Invalid email format"
2) "O formato do campo e-mail está inválido" -> "The email field has an invalid format"
*/
Alerta.popupAlertaComBotaoOK("Invalid email format", "The email field has an invalid format",
LoginActivity.this);
progressDialog.dismiss();
return null;
}
}
List<NameValuePair> listaParametros = new ArrayList<NameValuePair>();
listaParametros.add(new BasicNameValuePair("login", sLogin));
listaParametros.add(new BasicNameValuePair("senha", sSenha));
/* Translation note: Original text: "valores recuperados dos campos de login e senha: " */
Log.d(TAG, "values retrieved from login and password fields:" + sLogin + " | " + sSenha);
/* Translation note: Original text: "Reutiliza cliente HttpClient disponibilizado pela Aplicação." */
// Reuses `HttpClient` made available by the Application.
AnototudoApp atapp = (AnototudoApp) LoginActivity.this.getApplication();
HttpClient httpClient = atapp.getHttpClient();
String result = null;
try {
result = HttpProxy.httpPost(AnototudoMetadata.URL_AUTENTICACAO, httpClient, listaParametros);
} catch (IOException e) {
Log.e(TAG, "IOException, Sem conectividade com o servidor do Anototudo! " + e.getMessage());
e.printStackTrace();
return result;
}
return result;
}
@Override
protected void onPostExecute(String result)
{
super.onPostExecute(result);
if (result == null || result.equals("")) {
progressDialog.dismiss();
/*
Translation notes:
1) "Dados incorretos" -> "Incorrect data"
2) "Os dados informados não foram encontrados no Sistema! Informe novamente ou cadastre-se antes pela internet." -> "The reported data was not found in the System! Please report again or sign up on the internet first."
*/
Alerta.popupAlertaComBotaoOK(
"Incorrect data",
"The reported data was not found in the System! Please report again or sign up on the internet first.",
LoginActivity.this);
return;
}
/* Translation note: Original text: "Login passou persistindo info de login local no device" */
Log.d(TAG, "Login passed persisting local login info on device");
ContentValues contentValues = new ContentValues();
contentValues.put(AnototudoMetadata.LOGIN_EMAIL, sLogin);
contentValues.put(AnototudoMetadata.LOGIN_SENHA, sSenha);
contentValues.put(AnototudoMetadata.LOGIN_SENHA_GERADA, result);
LoginDB loginDB = new LoginDB();
loginDB.addLogin(LoginActivity.this, contentValues);
/* Translation note: Original text: "Persistiu info de login no device, redirecionando para menu principal do Anototudo" */
Log.d(TAG, "Persisting login info on device, redirecting to Anototudo main menu");
/* Translation note: Original text: "O retorno da chamada foi ==>> " */
Log.d(TAG, "The callback was ==>>" + result);
/* Translation note: Original text: "tudo ok chama menu principal" */
// everything ok call main menu
/* Translation note: Original text: "Device foi corretametne autenticado, chamando tela do menu principal do Anototudo." */
Log.d(TAG, "Device has been correctly authenticated by calling the main menu screen of Annotate.");
String actionName = "br.com.anototudo.intent.action.MainMenuView";
Intent intent = new Intent(actionName);
LoginActivity.this.startActivity(intent);
progressDialog.dismiss();
}
}
期待どおりに動作するようになりましたが、AsyncTask
のドキュメントでは、executeを使用してタスクにパラメーターを渡すことができると言われているので、少し混乱していると言わざるを得ません。