В написании кода помогли следующие статьи (кои стоит прочитать):
http://startandroid.ru/ru/uroki/vse-uroki-spiskom/62-urok-25-task-chto-eto-takoe-i-kak-formiruetsja.html
http://startandroid.ru/ru/uroki/vse-uroki-spiskom/190-urok-116-povedenie-activity-v-task-intent-flagi-launchmode-affinity
http://stackoverflow.com/questions/9839592/getting-a-list-of-running-processes-and-killing-a-specific-process
http://stackoverflow.com/questions/7992563/how-to-find-back-stack-activities-in-an-android-application
и статейка о работе оператора if в java
http://developer.alexanderklimov.ru/android/java/if.php
Приложение все тоже. В нем есть три Активности – ActivityA, ActivityB и ActivityC. На каждой Активности есть кнопка вызывающая следующую активность. A->B->C->A и так далее. Сейчас добавил счетчик количества активностей в задаче а так же вывод информации о том была ли уже запущена Активность или нет.
Итак сперва запустим приложение, чтобы было понятно о чем речь, потом рассмотрим код.
Запустили и видим что Активность нам сообщает, (1) что этот экземпляр Активности запущен в первый раз. А так же видно количество Активностей в текущей задаче (2). У нас пока одна Активность, так как кнопку Start Activity B мы не нажимали.
Теперь посмотрим логи
Здесь все стандартно. Все как на классических диаграммах.
Далее жмем кнопку Home (3). Наше приложение уходит в фоновый режим. Смотрим логи
Приложение перешло в остановленное состояние.
Теперь заново откроем его тапнув по его иконке в списке приложений
Теперь Активность сообщает что она была уже запущена и просто восстановлена из памяти.
Из логов видно, что события onCreate для Активности не было, и она не создавалась заново, а была просто возобновлена ее работа методом onRestart.
Теперь нажмем кнопку Start Activity B
Мы видим что Активность В была запущена впервые и что в нашей задаче уже две активности А и В.
Посмотрим логи
Мы видим что после нажатия на кнопку Start Activity B, сперва Активность А перешла в приостановленное состояние onPause и только потом создалась и отобразилась Активность В. И уже после этого Активность А перешла в остановленное состояние onStop. Это один из важных моментов для понимания.
Теперь нажмем кнопку Обратно (Back), чтобы вернуться к Активности А и посмотрим на нее и логи
Сейчас Активность А сообщает, что она уже была запущена прежде, а Активностей в нашей задаче снова одна, так как Активность В была уничтожена при нажатии в ней кнопки обратно, что подтверждают и логи
Теперь снова запустим Activity B, а из нее Activity C
Комментировать особо не буду, тут и так все понятно по аналогии с предыдущим. Просто приведу еще логи выполнения.
Сейчас у нас Активности А и В находятся в остановленном состоянии, но они существуют. Количество Активностей в нашей задаче равно 3 – А, В и С.
Теперь запустим из Активности С Активность А. Это будет НОВЫЙ экземпляр Актиновости А, другая Активность А, с которой мы начинали сейчас находится на дне стека.
Собственно скрин это и показывает. Теперь в нашей задаче 4 активности А, В, С и снова А. Теперь если нажать кнопку HOME. И снова вернуться к приложению, то мы увидим, что новая Активность А уже была запущена.
И логи
Видно что Активность А была приостановлена, затем остановлена и затем возобновила свое состояние.
Теперь если мы начнем последовательно нажимать кнопку ОБРАТНО, то очистим стек
Активностей и выйдем из приложения.
По существу этот материал как раз продемонстрировал на практике, то что было изложено в этой моей статье.
Теперь перейдем к коду этого приложения.
В приложении три Активности и три файла разметки. Я приведу код ActivityA.java и layout_a.xml, а так же код AndroidManifest.xml. Файлы остальных Активностей и разметок аналогичны.
И так ActivityA.java
package com.example.ap0003;
import java.util.List;
import android.app.Activity;
import android.app.ActivityManager;
import android.app.ActivityManager.RunningTaskInfo;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.TextView;
public class ActivityA extends Activity {
final String TAG = "States";
TextView tvTextLife;
List<ActivityManager.RunningTaskInfo> list;
ActivityManager am;
Integer TotalActCount;
Boolean FirstStart;
Boolean NextAct;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.layout_a);
Log.d(TAG, "ActivityA: onCreate()");
// флаг что активность запущена впервые
FirstStart = true;
// кнопка запуска следующей Активности не нажималась
NextAct = false;
}
@Override
protected void onStart() {
super.onStart();
Log.d(TAG, "ActivityA: onStart()");
}
@Override
protected void onResume() {
super.onResume();
Log.d(TAG, "ActivityA: onResume()");
// получаем список 10 последних задач
am = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
list = am.getRunningTasks(10);
// перебираем список задач и выбираем свою по имени пакета
// com.example.ap0003
for (RunningTaskInfo task : list) {
if (task.baseActivity.flattenToShortString().startsWith(
"com.example.ap0003")) {
// находим поле для вывода информации о количестве запущенных
// Активностей
tvTextLife = (TextView) findViewById(R.id.textActCountA);
TotalActCount = task.numActivities;
// коррекция счетчика для кнопки ОБРАТНО
if (NextAct == true & FirstStart == false)
TotalActCount = TotalActCount - 1;
// выводим количество Активностей в задаче
tvTextLife.setText("Activites in task " + TotalActCount);
// коррекция счетчика для кнопки ДОМОЙ
NextAct = false;
}
}
}
@Override
protected void onPause() {
super.onPause();
Log.d(TAG, "ActivityA: oPause()");
// флаг что активность уже была запущена
FirstStart = false;
// находим текстовое поле по его идентификатору
tvTextLife = (TextView) findViewById(R.id.textStateActA);
// присваиваем значение атрибуту Text для выбранного TextView
tvTextLife.setText("Этот экземпляр ActivityA уже был запущен!");
}
@Override
protected void onStop() {
super.onStop();
Log.d(TAG, "ActivityA: onStop()");
}
@Override
protected void onRestart() {
super.onRestart();
Log.d(TAG, "ActivityA: onRestart()");
}
@Override
protected void onDestroy() {
super.onDestroy();
Log.d(TAG, "ActivityA: onDestroy()");
}
public void onClickStartB(View v) {
Intent intent = new Intent(ActivityA.this, ActivityB.class);
startActivity(intent);
// кнопка запуска следующей Активности была нажата
NextAct = true;
}
}
Обратите внимание на выделенные строки. Это я задал флаги состояния и их проверку, чтобы правильно скорректировать значение счетчика Активностей в задаче. Если этого не сделать, то поскольку, например, при нажатии кнопки ОБРАТНО восстанавливается предыдущая Активность, а та с которой был переход еще не уничтожена, то значение счетчика будет показывать на одну активность больше чем есть на самом деле. Тоже самое и с кнопкой Домой.
Теперь код layout_a.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#f20808"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".ActivityA" >
<TextView
android:id="@+id/textView1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/strA" />
<Button
android:id="@+id/buttonStartB"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_below="@+id/textView1"
android:onClick="onClickStartB"
android:text="Start Activity B" />
<TextView
android:id="@+id/textStateActA"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/buttonStartB"
android:layout_below="@+id/buttonStartB"
android:background="#fffcfc"
android:text="Этот экземпляр АсtivityA запущен в первый раз" />
<TextView
android:id="@+id/textActCountA"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/textStateActA"
android:layout_below="@+id/textStateActA"
android:layout_marginTop="15dp"
android:background="#fffcfc"
android:text="TextView" />
</RelativeLayout>
Выделенная строка 24 показывает вызов метода из ActivityA по нажатию кнопки.
И теперь код AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.ap0003"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk
android:minSdkVersion="8"
android:targetSdkVersion="19" />
<uses-permission android:name="android.permission.GET_TASKS"/>
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.example.ap0003.ActivityA"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name="com.example.ap0003.ActivityB"
android:label="@string/app_name" >
</activity>
<activity
android:name="com.example.ap0003.ActivityC"
android:label="@string/app_name" >
</activity>
</application>
</manifest>
Строка 10 добавляет разрешение для получения информации о задачах приложением. Если эту сроку не добавить, то приложение будет завершаться с ошибкой, так как у него не будет полномочий на это.
На этом пока закончим эту часть.
Комментариев нет:
Отправить комментарий