В написании кода помогли следующие статьи (кои стоит прочитать):
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 добавляет разрешение для получения информации о задачах приложением. Если эту сроку не добавить, то приложение будет завершаться с ошибкой, так как у него не будет полномочий на это.
На этом пока закончим эту часть.
Комментариев нет:
Отправить комментарий