Это вторая часть книги, здесь рассмотрены: уведомления — Notifications, потоки и асинхронное выполнение задач — Threads & AsyncTask, работа с сетью, приемники широковещательных сообщений — Broadcast Receivers, оповещения — Alarms, графика и анимация, управление тачем и жестами, управление мультимедией — музыкой, видео и встроенной камерой, работа с датчиками, определение местоположения и привязка к картам, управление данными, а также классы ContentProvider и Service.
Часть 1 содержит шесть глав, описывающих основные принципы создания приложений, пользовательский интерфейс, полномочия приложений, а так же базовые классы: Activity, Intent, Fragment.
Книга предназначена для программистов, владеющих языком программирования Java и желающих освоить написание приложений, работающих под ОС Android. Книга является переводом общедоступных бесплатных англоязычных интернет ресурсов.
Notifications — Уведомления
Уведомление ― это сообщение, которое может быть выведено на экран за пределами обычного пользовательского интерфейса приложения. Например, предположим, что у вас есть приложение, которое может загрузить электронную книгу из Интернета. Во время загрузки пользователь может продолжить использовать приложение или даже выйти из него в то время, как книга загружается. И тогда вы, вероятно, захотите сообщить пользователю об окончании загрузки. Чтобы сделать это, вы должны выяснить, когда загрузка закончится, а затем вывести на экран сообщение об этом.
Далее рассмотрим два различных вида пользовательских уведомлений, которые поддерживает Android. Первый вид сообщения — тост (toast). Тост обеспечивает простую обратную связь от операции в небольшом всплывающем окне. Это окно занимает небольшое пространство, минимально необходимое для сообщения, а текущая Activity остается видимой и интерактивной.
Второй вид — уведомления в строке состояния (status bar) или области уведомлений. Когда приложение сообщает системе о необходимости выдать уведомление, оно сначала отображается в виде значка в области уведомлений в верхней части экрана слева.
Чтобы просмотреть подробные сведения об уведомлении, пользователь открывает панель уведомлений, «вытягивая» ее сверху.
Строкой состояния и панелью уведомлений управляет система, а пользователь может их просматривать в любое время. Такие уведомления обычно используются, чтобы объявить о непредсказуемых событиях и сделать это способом, который не прерывает текущую деятельность пользователя независимо от того, что он делает в этот момент. Например, о входящей смс. Появление значка в области уведомлений говорит о том, что сообщение прибыло. Но это делается ненавязчивым способом — сообщение здесь, и вы свободны прочесть его тогда, когда это удобно для вас.
Итак, Android обеспечивает несколько различных видов пользовательских уведомлений. Эти уведомления — сообщения пользователю, которые отправляются по мере необходимости, и появляются они вне пользовательского интерфейса приложения. То есть и текст смс, который вы редактируете, и кнопка, которую вы нажмете, чтобы отправить сообщение, всегда видимы, пока вы сочиняете сообщение.
Тост-сообщения являются кратковременными сообщениями, которые разворачиваются на дисплее, например, чтобы дать пользователю знать об успешном завершении операции. Тосты автоматически постепенно появляются и постепенно исчезают. Их задача — предоставить информацию пользователю. Но они не предназначены для того, чтобы собирать информацию для передачи её обратно в приложение.
Для создания тоста, используется метод makeText класса Toast. Этот метод имеет два параметра: текст, который вы хотите вывести на экран, и количество времени, в течение которого текст должен быть видимым. После того, как вы создали тост, вы можете вывести его на экран, вызвав метод Toast.show ().
Теперь давайте посмотрим на пример приложения, которое использует тост-сообщения. Это приложение выводит на экран единственную кнопку, подписанную «Show Toast». Если нажать эту кнопку, то в нижней части экрана вы увидите маленькое раскрывающееся окно, которое говорит: «You’re toast!».
Теперь, если открыть исходный код приложения, мы увидим, как это реализовано.
Здесь вы видите OnClickListener для кнопки «Show Toast», а внутри — вызов метода makeText, передающий текст и константу Toast. LENGTH_LONG, которая делает текст видимым в течение приблизительно трех с половиной секунд. И в конце строки происходит вызов метода show (), который и выводит на экран тост.
Если вам не нравится стандартный вид тоста, вы можете создать кастом вью (пользовательское вью) для своего тоста. Например, вы можете создать кастом лейаут в XML и применить его, а затем присоединить созданную вью к тост-сообщению, вызвав метод setView.
Давайте посмотрим пример. Он во многом выглядит как предыдущий, но с той разницей, что здесь на экран выводится тост, созданный из пользовательской кастом вью.
Теперь давайте откроем основную Activity этого приложения и посмотрим, как был создан этот тост. Видно, что когда нажимается кнопка «Show Toast», код сначала создает новый объект тост-сообщения.
Следующие две строки устанавливают расположение тоста на экране и определяют отрезок времени, в течение которого сообщение будет видимым. Далее идет вызов setView, в котором первый параметр — результат применения XML-лейаута, который находится в пользовательском файле custom_toast. xml. В этом файле находится relativeLayout, содержащий два дочерних элемента. Первый — вью изображения, которое вы видите в тосте рядом с текстом. Второй — текстовое вью, которое выводит на экран текст «You’re toast!».
Затем, возвращаясь назад в основную Activity, в заключительной строке мы видим метод show (), который выводит созданный тост на экран.
Другой вид пользовательского уведомления, который мы рассмотрим — это уведомления, которые появляются в области уведомлений (status bar). Приложения и сама система Android могут использовать эту область, чтобы сообщить пользователю о возникновении различных событий. Область уведомлений также предоставляет такой элемент пользовательского интерфейса, как панель (drawer), которую пользователь может открыть, вытянув из области уведомлений сверху. И если её открыть, вы увидите дополнительную информацию о различных уведомлениях, которые были помещены в область уведомлений.
Давайте рассмотрим пример того, как эти уведомления используются в Android. Откроем приложение Телефон и наберем телефонный номер. Телефон начнет вызов и соединит с вызываемым абонентом. Теперь, если посреди этого телефонного вызова, понадобится получить некоторую информацию из интернета, можно нажать кнопку «Домой», чтобы вернуться к домашнему экрану и оттуда открыть приложение браузера. В верхнем левом углу экрана устройства, появится новый значок, это — уведомление.
Когда пользователь отлучился из телефонного приложения, Android создал объект уведомления и поместил его в область уведомлений. Это уведомление служит напоминанием, что телефонное соединение все еще активно, а также служит способом быстро вернуться к этому телефонному вызову. Но в это время пользователь запустил браузер, где перешёл на www.google.com и выполняет поиск. Затем вооруженный информацией, в которой нуждался, он хочет вернуться к телефонному вызову. Тогда он вытягивает панель из области уведомлений, чтобы увидеть вьюшку, которая содержит некоторую информацию о вызове. Это позволяет ему либо вернуться к разговору, либо повесить трубку. Если он хочет продолжить разговор, то он кликнет по области уведомлений, и это переведёт телефонное приложение в рабочее состояние, вернет его на передний план и позволит пользователю продолжать разговор.
Если ваше приложение должно отправлять уведомления, вам придется предусмотреть несколько вещей. Во-первых, само уведомление, у которого должны быть, по крайней мере, заголовок, текст содержимого, и маленький значок — иконка. Когда уведомление будет отправлено, оно в конечном счете появится в области уведомлений, где эта иконка и будет выведена на экран. Кроме того, вы можете установить текст тикера уведомления, тогда этот текст будет также выведен на экран вместе с иконкой в области уведомлений. Наконец, если пользователь открывает панель уведомлений, должна быть вьюшка, которую пользователь и будет видеть в раскрытой панели. Далее вы должны будете определить какое-либо действие, которое произойдет, когда пользователь кликнет по уведомлению, вытянув панель уведомлений.
Теперь, когда вы создали уведомление, в какой-то момент вы захотите отправить его, затем обновить, отменить или что-то еще. Этими операциями управляет системная служба Android, называемая Менеджером уведомлений.
Давайте разберем два приложения, которые отправляют уведомления, и рассмотрим их исходный код, чтобы увидеть, как все это реализовано. Запустим приложение NotificationStatusBar. Пользовательский интерфейс содержит на экране единственную кнопку — «Notify». Если нажать эту кнопку, будет создано и отправлено уведомление и, в конечном счете, оно появится в панели уведомлений наверху экрана телефона.
Итак, при нажатии на кнопку прозвучал звук, присоединенный к уведомлению и в верхней части экрана появилась бегущая строка с текстом уведомления. По окончанию прокручивания бегущей строки текст пропадает, а на его месте остается иконка уведомления.
А теперь откроем панель уведомлений. Вы видите, что вьюшка уведомления показывает иконку, текст заголовка уведомления, подробный текст уведомления, который показывает номер один в круглых скобках, указывая, что кнопка «Notify» была нажата один раз. И имеется также метка времени. Если закрыть панель уведомлений и нажать кнопку «Notify» еще раз, то текст уведомления будет обновлен, чтобы показать, что кнопка опять была нажата.
Далее, если кликнуть по самому уведомлению, вы увидите, что запустилась новая Activity, напечатав слова: «Got the Intent». Смысл в том, что вы можете присоединить Intent к вьюшке в панели уведомлений, чтобы переключить пользователя на приложение (или Activity), которое должно обработать то действие, о котором говорилось в уведомлении.
Второе приложение об уведомлениях в панели уведомлений с использованием кастомных вью делает то же самое, что и предыдущий пример, только показывает пользовательскую вьюшку, когда открыта панель уведомлений. Здесь также есть изображение глаза и слова, о том, что вы были уведомлены с числом в круглых скобках. И наконец, если кликнуть по этой вью, то будет запущено новое Activity, выводящее на экран слова «Got the Intent».
Теперь давайте рассмотрим код приложения с кастомной вью. Откроем основную Activity приложения.
Начиная сверху, код создает ID для уведомления, которое будет отправлено. Это позволит менеджеру уведомлений обновлять это уведомление, после того, как оно было отправлено. И затем, имеются некоторые переменные, которые содержат текстовые элементы уведомления, включая его текст тиккера (бегущей строки), заголовок и содержание. После этого, код делает некоторые установки, которые используются, чтобы проиграть звук и включать вибрацию, когда прибывает уведомление.
Затем, код создает пользовательскую кастом вью, которая будет выведена на экран в панели уведомлений.
Лейаут этой вью находится в файле custom_notification. xml. Эта вью представляет собой linear layout с двумя дочерними вью. Первый — изображение глаза. А другой — текстовая вью, которая выводит на экран текст «You’ve been notified».
Далее в основной Activity идет метод onCreate. Здесь код создает Intent, названный mNotificationIntent, который явно активирует sub-Activity уведомления.
Следующая строка кода это неописанный прежде объект — PendingIntent, который основан на известном mNotificationIntent, созданный на предыдущей строке. PendingIntent описывает интент (Intent) и действие, которое надо с ним выполнить, позволяет стороннему приложению выполнять определенный код вашего приложения с правами, которые определены для вашего же приложения. То есть позволяет стороннему приложению, в которое его передали, запустить хранящийся внутри него интент, от имени того приложения (и теми же с полномочиями), передавшего этот PendingIntent.
Идем дальше, «слушатель» кнопки «Notify» сначала обновляет текст содержимого, который указывает число нажатий кнопки.
Затем он создает само уведомление, используя класс notification. builder. Код создает новый объект notification. Builder, устанавливает текст тиккера (бегущей строки), устанавливает маленькую иконку и затем устанавливает параметр auto cancel равным true. Это укажет системе отменить уведомление, когда пользователь кликнет по вьюшке в панели уведомлений. Затем устанавливается интент, тот самый PendingIntent, который определяет какое действие произвести, когда пользователь кликнет по вьюшке в панели уведомлений. Далее устанавливаются звуки и образцы вибрации, которые должны быть воспроизведены, когда прибывает уведомление. И наконец, устанавливается пользовательская вью, которая должна быть выведена на экран, когда пользователь раскрывает панель уведомлений.
Теперь, когда уведомление установлено, код получает ссылку на менеджера уведомлений вызовом getSystemService, передавая ID службы уведомлений. И наконец, код вызывает метод notify менеджера уведомлений, передавая ID уведомления в качестве параметра.
А метод build непосредственно и генерирует тот самый объект уведомления.
Потоки
Мобильные устройства, как и компьютеры, сегодня все чаще содержат по несколько вычислительных ядер. Это означает, что несколько программ могут работать на этих устройствах одновременно. И это — мощная вещь, потому что она позволяет производить больше вычислений за меньшее время. Но это также сделает ваши программы намного более сложными, что может привести к ошибкам и проблемам производительности, если вы, как разработчик, не будете внимательны и осторожны.
Для одновременно работающих программ и параллельно производимых вычислений используются Потоки — Threads. Концептуально поток — одно из многих возможных вычислений, работающих одновременно в операционной системе. С точки зрения реализации каждый поток имеет свой собственный счетчик команд и стек времени исполнения, но совместно использует «кучу» (структура данных, с помощью которой реализована динамически распределяемая память приложения) и области статического ЗУ с другими работающими потоками.
Диаграмма, изображающая эти понятия.
Здесь показано гипотетическое вычислительное устройство. У этого устройства есть два процессора, CPU 1 и CPU 2. Каждый из этих процессоров может выполнять инструкции приложений, работающих на устройстве. Далее на CPU 2 показаны два выполняющихся процесса — p3 и p4. Один из вариантов — рассматривать процессы как автономные среды исполнения. У них есть ресурсы, такие как память, открытые файлы, сетевые соединения и другие вещи, которыми они управляют и разделяют с другими процессами на устройстве. И в одном из этих процессов, p4, показаны два работающих потока — т7 и т8. Каждый из этих потоков — последовательно выполняющийся ряд инструкций со своим собственным стеком вызовов. Но так как они работают в одном и том же процессе, каждый из них может получить доступ к совместно используемым ресурсам этого процесса, включая статические переменные.
В Java потоки представлены объектом типа thread в пакете java.lang. Потоки в Java реализуют интерфейс runnable. Это означает, что у них должен быть открытый (public) метод называющийся run, который не получает параметров и у него нет возвращаемого значения. Среди методов потоки имеют метод start для запуска потока и метод sleep для того, чтобы временно приостановить выполнение потока. Метод wait заставляет поток приостановиться и ждать пока другой поток не разбудит его методом notify.
Чтобы использовать поток, во-первых, необходимо создать поток. Например, при помощи команды new. Далее, потоки автоматически не запускаются, чтобы запустить поток необходимо вызвать метод start. И, в конечном счете происходит вызов метода run, который будет выполняться до тех пор, пока не будет остановлен.
Работающее приложение дает команду new, чтобы создать объект — новый поток. Далее работа приложения продолжается, и когда-то позже произойдет вызов метода запуска потока start. При этом произойдет возврат к приложению, но параллельно начнет выполняться код в методе потока run. И, поскольку программа продолжается, то теперь одновременно выполняются два потока. Таким образом, проделав это многократно, можно создать и выполнять столько потоков, сколько необходимо.
Рассмотрим приложение, в котором поточная обработка была бы полезна, ThreadingNoThreading. Приложение выводит на экран простой пользовательский интерфейс с двумя кнопками. Первая кнопка маркирована «LoadIcon». Когда пользователь кликает по этой кнопке, приложение открывает и читает файл, содержащий изображение, и затем показывает его на дисплее. Идея здесь состоит в том, что эта работа может занимать значительное количество времени. В данном примере количество времени несколько преувеличено для наглядности. Некоторые операции занимают относительно большое количество времени. И разработчик должен это понимать и правильно манипулировать этим.
Вторая кнопка маркирована «Other Button». Когда пользователь кликает по этой кнопке, раскрывается тост-сообщение, выводя на экран некоторый текст. А здесь идея в том, что, если вы видите текст, значит кнопка работает. Соответственно, если вы не можете нажать кнопку или не видите текст, значит что-то работает неправильно.
В идеале пользователь должен быть в состоянии нажать любую из кнопок в любое время и получить реакцию, система просто должна работать. Давайте запустим версию этого приложения, которое не использует поточной обработки. Нажмем кнопку «Other Button», и на дисплее появляется обещанное сообщение.
А теперь поступим иначе. Сначала нажмем кнопку «Load Icon», которая запустит трудоемкую работу чтения битового массива из файла и вывода изображения на экран. И сразу же после нажатия кнопки «Load Icon», не дожидаясь появления картинки, нажмем кнопку «Other Button». И что же тогда произойдет? Кажется кнопка «Other Button» не работает. Почему? Потому что при попытке нажать «Other Button», Android все еще загружал картинку после нажатия кнопки «Load Icon» и это препятствовало тому, чтобы вторая кнопка сработала.
У этой проблемы есть одно на вид очевидное, но, в конечном счете, неправильное решение — перейти к «слушателю» кнопки «Load Icon» и просто создать новый поток, который будет загружать битовый массив и затем выведет его на экран. Пример такой реализации — приложение Threadingsimple, вот его код.
«Слушатель» кнопки «Load Icon» вызывает метод loadIcon (), который показан чуть ниже. Этот код создает новый поток, который занимает время, загружая битовый массив. И затем пытается установить изображение в image view, который является частью лейаута.
Теперь, если запустить приложение и нажать кнопку «Load Icon», а затем сразу нажать «Other Button», видно, что есть реакция на «Other Button» и её нажатие не блокируется. Это хорошо, есть прогресс. Однако появилась большая проблема, через несколько секунд приложение рухнуло.
Если исследовать журнал logcat, мы увидим, что есть сообщение о том, что только тот поток, в котором создана иерархия вьюшек, может манипулировать своими вьюшками.
Это означает, что новый поток, который мы создали, чтобы загрузить битовый массив, может сделать эту работу, но он не сможет сделать последний шаг и вывести получившееся изображение на дисплей.
И так, какой же поток на самом деле создал иерархию вьюшек этого приложения? У всех Android приложений есть основной поток, который также называют UI thread — потоком пользовательского интерфейса. Все компоненты приложения, которые работают в одном и том же процессе, запущенном по умолчанию, используют тот же поток — UI thread. Все методы жизненного цикла: OnCreate, OnStart и т.д., работают в основном потоке. И, кроме того, сам инструментарий пользовательского интерфейса не ориентирован на многопотоковое исполнение. Все это означает, что, если вы заблокируете UI-поток любой продолжительной работой, это воспрепятствует реакции приложения на другие манипуляции пользователя. Следовательно, долго выполняющиеся операции должны быть помещены в фоновые потоки. Однако, вместе с этим, мы не сможем получить доступ к инструментарию пользовательского интерфейса от потока, не принадлежащего UI tread. Таким образом, мы должны делать длительную работу в фоновом потоке, но когда эта работа сделана, мы должны сделать обновление пользовательского интерфейса в потоке UI.
И Android на самом деле дает нам набор средств сделать это. В частности Android обеспечивает несколько методов, которые гарантированно будут работать в потоке пользовательского интерфейса. Два из них — это метод post класса View и runOnUiThread класса Activity. Оба этих метода получают параметр Runnable, содержащий код, который, например, обновляет экран в наших недавних примерах. Таким образом, если мы используем эти методы, мы могли бы загрузить битовый массив в фоновом потоке, и когда эта операция завершится, мы использовали бы один из этих методов, чтобы выполнить Runnable, который и выведет изображение на экран.
Давайте рассмотрим, как это реализуется в коде. Откроем основную Activity этого приложения и перейдем прямо к методу loadIcon, который вызывается, когда пользователь нажимает кнопку «Load Icon». Как и в предыдущем примере, этот код создает новый поток и затем загружает битовый массив. Но после загрузки данных, теперь происходит вызов view.post, запускающий новый Runnable, код которого и вызывает метод setImageBitmap, чтобы установить загруженное изображение в соответствующий imageView.
Следующий класс, обеспечивающий поточную обработку, это класс AsyncTask. Он обеспечивает правильное и простое использование потока пользовательского интерфейса. Этот класс служит общей основой для управления задачами, которые, как в наших предыдущих примерах, включают операции, которые должны выполняться в фоновом потоке, а их результат публикуется в UI потоке. Основой использования AsyncTask является то, что работа разделена между фоновым потоком и потоком UI. Фоновый поток выполняет длительную операцию и может дополнительно сообщать о ее прогрессе. Поток пользовательского интерфейса, с другой стороны, ответственен за начальную настройку длительно выполняемой операции, за публикацию промежуточной информации о прогрессе и за завершение операции после того, как фоновый поток сделал свою работу.
AsyncTask — универсальный класс. Он определяется тремя основными параметрами: params, progress, и result. Params — тип параметров, которые отправляются в AsyncTask при выполнении. Progress — тип любых единиц прогресса, публикуемого в процессе выполнения фоновой операции. Result — тип результата фонового вычисления.
Во время выполнения AsyncTask проходит через четыре шага. Во-первых, onPreExecute () — метод вызывается в потоке UI до начала работы фонового потока.
Сразу же после него запускается doInBackground (Params…) и выполняет вычисления в фоновом режиме. Сюда же передаются параметры. И этот же метод возвращает в последний шаг результат типа Result.
Во время работы doInBackground может дополнительно вызвать метод publishProgress (Progress…) с перечнем переменных, который обеспечит некоторую индикацию прогресса длительного процесса. А переданные переменные используются в UI потоке при помощи метода onProgressUpdate (Progress…), который запускается в потоке пользовательского интерфейса после вызова publishProgress.
И последний шаг, onPostExecute (Result) вызывается в UI потоке после окончания фоновых вычислений с результатом, возвращенным как его параметр.
Давайте рассмотрим версию нашего приложения загрузки картинки, реализованного с AsyncTask. Если мы запустим приложение, то увидим, что оно выглядит подобно предыдущим примерам, но здесь добавлен новый элемент интерфейса — индикатор выполнения, который показывает, какое количество работы по загрузке изображения уже выполнено.
После нажатия кнопки «Load Icon» появится маленький индикатор выполнения и начнет медленно заполняться. После нажатия второй кнопки раскроется знакомый текст — кнопка «Other Button» работает. И наконец, на экране появится изображение.
А теперь рассмотрим исходный код этого приложения. Откроем основную Activity.
«Слушатель» кнопки «Load Button» создает новый экземпляр LoadIconTask и сразу же происходит вызов execute с передачей в качестве параметра id ресурса изображения.
Далее рассмотрим класс LoadIconTask более подробно. LoadIconTask — это AsyncTask с параметрами: params — типа целое, progress — типа целое и result — типа битовый массив.
Первый метод, который мы рассмотрим, это onPreExecute. Этот метод выполняется в UI потоке и его назначение — сделать индикатор выполнения видимым на дисплее.
Следующий метод — doInBackground. Этот метод получает в качестве параметра целое число — id ресурса для битового массива, который был передан в метод LoadiconTask. execute. doInBackground выполняет работу загрузки битового массива. И пока он делает это, то периодически вызывает publishProgress, передавая параметр, представляющий собой процент загрузки, которая была проделана до этого момента. Этот пример является немного надуманным с целью упрощения понимания.
Следующий метод — onProgressUpdate. Этот метод выполняется в потоке пользовательского интерфейса, получает параметр, который был передан в PublishProgress, и затем устанавливает индикатор выполнения для отображения процента выполненной работы.
И наконец, последний метод — onPostExecute. Этот метод так же работает в UI потоке и получает только что загруженный массив (Bitmap) в качестве параметра.
Сначала onPostExecute делает индикатор выполнения невидимым, так как он больше не нужен и затем выводит загруженное изображение на экран.
Класс Handler — это обработчик, позволяющий отправлять и обрабатывать сообщения и объекты Runnable, ассоциированные с очередью сообщений потока. Каждый его экземпляр связан с отдельным потоком и его очередью сообщений. Как и AsyncTask, класс Handler предназначен для взаимодействия между двумя потоками, но он более гибок, так как может работать с любыми двумя потоками, а не только между фоновым и потоком пользовательского интерфейса. Каждый поток может передать работу другому потоку путем отправки сообщения или объекта Runnable обработчику, который с ним ассоциирован.
Мы используем объекты Runnable, когда отправитель (поток-заказчик) точно знает какую работу и как необходимо выполнить, но выполнить её надо в потоке обработчика. С другой стороны, Сообщение — это класс, который может содержать данные, такие как код сообщения, произвольный объект данных или целочисленные аргументы. И мы будем использовать сообщения, когда поток-отправитель указывает какую операцию необходимо выполнить в другом потоке, а как её выполнить определит handler.
В Android каждый поток связан с очередью сообщений — messageQueue и петлителем — Looper. Очередь сообщений — структура данных, которая содержит сообщения и объекты runnable. Петлитель забирает эти сообщения и объекты runnable из очереди сообщений и диспетчеризирует их в зависимости от обстоятельств.
Эта диаграмма изображает поток Tread A, который создает Runnable внутри метода post (опубликовать) и использует объект-обработчик — Handler, чтобы отправить его в поток этого обработчика. Runnable помещается в очередь сообщений потока, связанного с обработчиком. И тоже самое происходит с сообщениями.
А эта диаграмма изображает поток Thread B, который создает сообщение и использует метод обработчика sendMessage, чтобы отправить это сообщение в поток обработчика. Сообщение помещается в очередь сообщений, связанную с этим обработчиком.
В то время, когда выполняются все вышеописанные операции, объект-петлитель просто «сидит» ожидая работы, которая появится в очереди сообщений. И когда эта работа появляется, петлитель реагирует одним из двух способов в зависимости от вида работы, которая только что появилась. Если эта работа — сообщение, петлитель обрабатывает это сообщение, вызывая метод обработчика handleMessage и передавая в нем само сообщение. Если же эта работа будет Runnable, то петлитель просто вызывает его метод run.
Существует несколько методов, которые позволяют запланировать выполнение работы в разное время. Например, можно использовать метод postAtTime, чтобы добавить runnable в очередь сообщений и выполнить его в определенное вами время. Метод postDelayed позволяет добавить runnable в очередь сообщений и выполнить его после указанной задержки.
Если вы хотите отправлять сообщения, сначала необходимо сообщение создать. Один из способов это сделать — использовать метод обработчика obtainMessage, который выдаст вам сообщение уже с установленным обработчиком. Можно также использовать метод obtain класса message.
Что касается runnable, то есть несколько методов, которые можно использовать, чтобы отправить сообщение. Метод sendMessage уже упоминался. Есть также его версия, которая позволяет поместить сообщение вперед в очереди сообщений, чтобы оно было выполнено как можно скорее. Есть метод sendMessageAtTime, чтобы поставить сообщение в очередь согласно требуемому времени. Есть также метод sendMessageDelayed, который поставит сообщение в очередь в текущее время плюс указанная задержка.
Давайте рассмотрим исходный код некоторых версий нашего рабочего примера, в котором были использованы обработчики. Вот приложение Threading handler runnable. Откроем основную Activity этого приложения. Во-первых, вы видите, что этот код создает новый обработчик. Этот обработчик создается в основном потоке пользовательского интерфейса. Таким образом, объекты runnable, которые получает этот обработчик, будут выполняться в UI потоке.
Далее вы видите «слушателя» кнопки «Load Icon». Когда пользователь нажимает кнопку «Load Icon», этот код создает и запускает новый поток, метод run которой определен runnable-объектом LoadIconTask.
Метод run класса LoadIconTask начинается, создавая новый объект Runnable, который при исполнении сделает видимой шкалу прогресса.
Далее происходит загрузка битового массива. И во время этой загрузки, периодически публикуется её прогресс методом setProgress другого runnable.
Далее создается новый Runnable, который устанавливает только что загруженный битовый массив на экран. И заканчивается все, созданием последнего Runnable, который делает шкалу прогресса невидимой.
А теперь рассмотрим вторую версию этого приложения, которое отправляет сообщения вместо runnable — ThreadingHandlerMessages. Откроем основную Activity этого приложения. Во-первых, этот код создает новый обработчик. И как в предыдущем примере, этот обработчик будет создаваться основным UI потоком. И соответственно работа, которую выполняет этот обработчик, будет выполняться в UI потоке. У этого обработчика есть метод handleMessage, в котором он реализует различные виды работ. Этот метод вначале проверяет код сообщения, содержащийся в этом сообщении, и затем выполняет действие, определенное для этого кода сообщения.
Например, если код — set_progress_bar_visibility, то будет установлено состояние видимости шкалы прогресса. Если код — progress_update, то будет установлено значение прогресса на шкале прогресса. Если код — set_bitmap, то этот код выводит битовый массив на дисплей.
Теперь перейдем к «слушателю» кнопки «Load Icon». Как и в предыдущих примерах, когда пользователь нажимает кнопку, этот код, создает и запускает новый поток, чей метод run определен runnable-объектом LoadIconTask.
Выполнение этого метода начинается при получении сообщения с кодом set_progress_bar_visibility с параметром, указывающим, что шкала прогресса должна стать видимой. Тогда это сообщение отправляется в обработчик, который и сделает шкалу прогресса видимой.
Далее происходит загрузка битового массива. И одновременно с этим, периодически обновляется прогресс, при помощи сообщения с кодом progress_update и с параметром, который указывает процент выполненной работы. Это будет результатом вызова обработчиком метода setProgress объекта progressBar.
Затем принимается и отправляется сообщение, чтобы вывести только что загруженное изображение на дисплей. И наконец, отправляется последнее сообщение, чтобы сделать шкалу прогресса невидимой.
Работа с сетью
Одной из определяющих характеристик современных мобильных устройств является то, что они могут обеспечить нас связью и сетевым подключением, то есть он-лайн, не привязывая нас к определенному местоположению. Смартфоны и планшеты комбинируют мощные процессоры с быстрым сетевым соединением по WiFi и сотовым сетям. Мобильные приложения должны уметь использовать эти сетевые возможности, чтобы обеспечить нас данными и предоставить доступ к службам. Android включает в себя поддержку множества сетевых классов при помощи пакетов Java.net, org. appache и android.net.
В текущей главе мы рассмотрим некоторые из этих классов, используя каждый из них для реализации одного и того же примера приложения. Это приложение взаимодействует с интернет-службой для получения информации о землетрясениях, произошедших в определенном географическом регионе. Чтобы заставить это приложение работать, код должен создать http-запрос, отправить его на сервер, получить результаты и затем отобразить эти результаты.
Для реализации этого, мы будем использовать три класса, это класс сокетов (Socket), класс HttpURLConnection и AndroidHttpClient.
Если запустить приложение, использующее класс socket, первоначально мы увидим одну кнопку с надписью «Load Data». И, если нажать эту кнопку, приложение отправит запрос HTTP GET на внешний сервер, и этот сервер ответит сложным текстом, содержащим запрошенные данные о землетрясениях.
Давайте рассмотрим исходный код, чтобы узнать, что нужно для получения этих данных. Откроем основную Activity этого приложения, и здесь мы видим «слушателя» кнопки загрузки данных. Когда эта кнопка нажата, приложение создает и затем выполняет AsyncTask с именем HttpGetTask.
Класс HttpGetTask сначала объявляет некоторые переменные, которые используются при создании запроса HttpGet.
Первым вызывается метод doInBackground. Этот метод создает новый сокет, который будет соединен с хостом api.geonames.org по стандартному http-порту: 80.
Затем код получает исходящий поток сокета, записывает в него http_get_command, и эта строка будет отправлена хосту, который интерпретирует его как запрос HTTP GET, а затем ответит, отправив обратно соответствующие данные ответа. Далее код получает входящий поток сокета, передавая его методу readStream, который считывает данные ответа из входящего потока сокета, и возвращает ответ в виде строки.
Затем эта строка передается методу onPostExecute, который выполняется в главном потоке, и который отображает ответ в текстовой вью.
Вернемся к приложению. Текст ответа включает в себя не только данные землетрясения, но и заголовки ответа HTTP. Но вы, возможно, не хотели бы показывать здесь этот избыточный текст, а хотели бы вывести на экран только данные землетрясения. Поэтому вы должны будете проанализировать ответ и извлечь только те данные, которые хотели бы вывести на экран. Кроме того, здесь отсутствует код обработки ошибок, который сделал бы это приложение более надежным. Использование сокетов — это компромиссный вариант, сходный с программированием низкого уровня. Вы можете записать в сокет все, что хотите, но взамен вы должны будете обрабатывать все многочисленные детали создания HTTP-запросов, прописать всю обработку ошибок и всю обработку HTTP-ответов.
В следующей реализации используется класс HttpURLConnection. Этот класс предоставляет интерфейс более высокого уровня, который обрабатывает больше деталей сетевого соединения, чем класс сокетов, но он также имеет менее гибкий API, чем другой вариант — класс Android HTTP client. Также необходимо отметить, что команда Android больше не работает над клиентом Android HTTP, отложив улучшения этого класса на будущее.
Итак, давайте рассмотрим пример приложения, реализованного на этот раз с классом HttpURLConnection. Приложение NetworkingURL внешне выглядит и работает также как и приложение из предыдущего примера, но после нажатия кнопки «Load Data» в текстовом выводе отсутствуют Http заголовки, они были удалены.
Давайте рассмотрим исходный код и посмотрим, как это работает. Откроем основную Activity этого приложения, и здесь мы увидим «слушателя» кнопки загрузки данных. Как и ранее, при нажатии этой кнопки приложение создаст и выполнит AsynchTask с именем httpGetTask.
Давайте рассмотрим этот класс. Когда в HttpGetTask срабатывает метод execute, вызывается метод doInBackground. Этот метод начинается с создания нового объекта URL и передачи строки URL-адреса для требуемой службы в качестве параметра.
Затем код вызывает метод openConnection для объекта URL, который возвращает Http соединение. Этот объект затем сохраняется в переменной httpUrlConnection.
Код продолжается, получая поток входящих данных (input stream) Http-соединения и передавая его в метод readStream. А метод readStream считывает данные ответа из сокета входящего потока, а затем возвращает ответ в виде строки. Однако на этот раз httpUrlConnection лишает заголовков http-ответ и обрабатывает проверку ошибок.
Далее эта строка передается методу onPostExecute, который выводит на экран ответ в текстовую вью.
Третий класс — AndroidHTTPClient. Этот класс является реализацией DefaultHttpClient проекта Apache. Он позволяет многое настроить. В частности, класс разбивает транзакцию HTTP на объект запроса и объект ответа. Это дает возможность создавать подклассы, которые настраивают обработку запросов и их ответов.
Пример приложения выглядит так же, поэтому перейдем прямо к коду и посмотрим на реализацию в приложении NetworkingAndroidHttpClient.
Откроем основную Activity этого приложения и перейдем к классу HttpGetTask. Этот класс начинается с создания нового объекта AndroidHttpClient, вызывая метод newInstance.
Далее вызывается метод doInBackground, код создает объект HttpGet, передавая в запросе URL-строку.
Затем создается объект ResponseHandler — обработчик ответа. В этом случае для запроса HttpGet обработчик ответа имеет тип BasicResponseHandler, который возвращает тело ответа.
И, наконец, запрос и обработчик ответа передаются в метод execute, который отправляет запрос, получает ответ, передавая его через обработчик ответа. Результат всего этого затем передается в onPostExecute, который выводит ответ в текстовую вью.
До сих пор примеры приложения запрашивали данные, а затем просто отображали эти данные в текстовой вью в том виде, в котором получили. Но эти данные имеют сложный формат и неудобны для восприятия человеку, поэтому нуждаются в дополнительной машинной обработке.
По сути, это все более популярный способ передачи текстовых данных через интернет, и многие веб-сервисы сейчас предоставляют данные в таких форматах. В частности, два формата, о которых мы будем говорить, это JavaScript Object Notation — JSON и Extensible Markup Language — XML. Рассмотрим каждый из них по отдельности.
Первый формат, это JavaScript Object Notation — JSON. Этот формат предназначен для небольших объемов и напоминает структуры данных, встречающиеся в традиционных языках программирования. Данные JSON упаковываются в два типа структур данных. Один — карты, которые являются наборами пар ключ — значение, и два — упорядоченные списки.
Теперь вернемся к примеру приложения, которое обращается к веб-службе за информацией о землетрясениях. Ответ, который вернулся, фактически был отформатирован в JSON.
Вот эти данные, давайте разобьем их на части.
Во-первых, данные содержат один объект JSON, и этот объект является картой, которая имеет одну пару ключ — значение. Ключ называется «earthquakes», а значение — это упорядоченный список. Этот список имеет несколько объектов, и каждый из этих объектов сам является картой, содержащей пары ключ — значение.
Например. Есть ключ под названием «eqid» и его значением является id землетрясения. Далее идет ключ «magnitude» с некоторым числовым значением. Затем ключ «lng», его значение — это долгота, на которой произошло землетрясение. Кроме того, есть множество других ключей, и все вместе эти значения являются данными для одного землетрясения.
Давайте посмотрим на пример приложения, которое получает эти данные из интернета, а затем обрабатывает их так, чтобы создать более читабельный для пользователя вид. Запустим приложение NetworkingAndroidHttpClientJSON. Как и ранее, это приложение первоначально отображает единственную кнопку с пометкой «Load Data» и, как и ранее, при нажатии этой кнопки приложение отправляет HTTP-запрос на внешний сервер, и этот сервер будет отвечать сложным текстом, содержащим запрошенные данные о землетрясениях. Однако на этот раз данные будут обработаны и представлены в виде списка.
Давайте рассмотрим исходный код, чтобы увидеть, как это работает. Сразу переходим к классу HttpGetTask.
Здесь метод doInBackground аналогичен тому, что мы видели раньше, но на этот раз он использует класс JSONResponseHandler для обработки ответа. Ключевым методом в этом классе является метод handleResponse. Этот метод начинается с передачи необработанного ответа через базовый обработчик ответа — BasicResponseHandler, который просто возвращает тело ответа без заголовков ответа HTTP.
Затем код использует JSONTokener для разбора JSON-ответа в объект Java, чтобы затем возвратить этот объекта верхнего уровня, который в данном случае является картой.
Затем код извлекает значение, связанное с ключом землетрясения. В этом случае это упорядоченный список.
Затем код перебирает список землетрясений. И для каждого элемента этого списка он получает данные, связанные с одним землетрясением, и эти данные сохранены в картах.
Затем код суммирует различные части данных землетрясения, преобразуя их в одну строку и добавляя эту строку в список с именем result.
И затем, наконец, результат возвращается обратно вызывающему методу.
Второй формат данных, который мы рассмотрим — это Extensible Markup Language — XML. XML — это язык разметки для создания XML-документов. XML-документы содержат разметку и контент. Разметка кодирует описание структуры хранения в документе и логической структуры при помощи тегов и атрибутов. Контент — это все остальное. И, в частности, контент содержит данные ответа, когда XML используется для кодирования ответа HTTP.
Теперь вернемся к примеру приложения. Если мы зададим немного другой URL, то веб-сервис вернет данные землетрясения в формате XML, а не в формате JSON. Итак, вот эти данные.
Вначале есть элемент — тег, называемый geonames. В этот элемент вложен ряд элементов землетрясения и каждый из элементов землетрясения содержит другие элементы, которые обеспечивают данные для одного землетрясения.
Подобно тому, что мы видели в формате JSON, есть элемент eqid, его значение является идентификатором землетрясения. Есть также элемент lng, его значение — долгота, на которой землетрясение произошло, и точно так же, как в примере JSON, есть и множество других элементов.
Таким образом, если приложение получает XML-данные из интернета, ему нужно будет разобрать XML-документ, чтобы создать список для вывода на экран. Для разбора XML-документов Android предоставляет несколько различных типов парсеров XML.
Парсер DOM — Document Object Model (объектный). Парсеры DOM читают весь XML-документ и преобразуют его в структуру объектной модели документа — дерево, а затем приложение обрабатывает эту древовидную структуру. Этот парсер требует больше памяти, но позволяет приложению делать многопроходную обработку документа.
SAX — Simple API for XML (событийные) парсеры читают XML-документ как поток. И когда они сталкиваются с новым тегом в документе, они производят возврат в приложение, которое и обрабатывает информацию в этом теге. Эти парсеры используют меньше памяти, чем DOM-парсеры, но они ограничены выполнением обработки за один проход документа.
Pull-парсеры, так же как и SAX-парсеры, читают документ как поток, но используют подход, основанный на итераторах, где приложение, а не парсер, решает, когда следует переходить к следующему шагу синтаксического анализа. Pull-парсеры также используют меньше памяти, чем DOM, но они в дополнение дают приложению больший контроль над процессом синтаксического анализа, чем SAX-парсеры.
Пример приложения выглядит точно так же, как тот, который мы рассмотрели при разборе ответов JSON. Поэтому перейдем к исходному коду этого приложения.
Рассмотрим сразу класс HTTPGetTask. Метод doInBackground похож на тот, что мы видели ранее. Но теперь он использует класс XML response handler для обработки ответа. Ключевым методом в этом классе является метод handleResponse, он начинается с создания объекта PullParser.
Затем код устанавливает вход парсера в качестве XML-документа, который был возвращен телом HTTP-ответа. После этого код получает первое событие парсера и затем начинает перебирать XML-документ.
Внутри цикла while есть 3 события, наличие которых этот код проверяет: стартовый XML-тег, конечный XML-тег и содержимое элемента. После определения какое событие наступило, происходит вызов соответствующего метода.
Вызывается метод startTag, в качестве параметра получая элемент, который начинается. Этот метод идентифицирует, является ли полученный элемент данных тем, который необходимо сохранить, и если это так, он сохраняет его, задавая значения определенным переменным.
Метод endTag получает в качестве параметра заканчивающий элемент. И этот метод так же определяет, является ли полученный элемент данных тем, который должен быть сохранен, и если это так, он его сохраняет. Кроме того, если это конечный тег землетрясения, тогда в список результатов добавляется результирующая строка для этой части данных землетрясения.
Когда событие является контентом, вызывается метод text. В качестве параметра передается содержимое элемента. Этот метод определяет, какой тег в настоящее время обрабатывается, и сохраняет содержимое для последующего использования.
После завершения метода doInBackground в основной Activity, как и в предыдущих примерах, вызывается метод onPostExecute c результатом, переданным в качестве параметра.
Этот метод создает и устанавливает адаптер списка для вывода на экран, передавая в качестве параметра результирующий список, который был вычислен при работе метода handleResponse.
И последнее. Чтобы ваше приложение могло работать с интернетом, ему необходимо предоставить соответствующее разрешение:
Broadcast Receivers — приемники широковещательных сообщений
BroadcastReceiver — базовый класс для кода, который получает и обрабатывает рассылки (сообщения, упакованные в Intent), отправленные методом sendBroadcast (Intent) для компонентов, целью которых является ожидание определенных событий, чтобы принимать эти события и реагировать на них. И способ, которым все это работает, состоит в том, что отдельные широковещательные приемники регистрируются для получения конкретных событий, в которых они заинтересованы.
Например, в Android есть широковещательный приемник, задачей которого является прослушивание входящих SMS-сообщений. Затем где-то в еще, некий компонент совершает какое-то действие, о котором он хочет сообщить вещательным приемникам, например отправить SMS. Тогда он создает интент, представляющий это событие, и передает этот интент, как будто в радиоэфир. Когда Android получит широковещательный интент, содержащий SMS-сообщение, он проверяет какие службы (или приложения) зарегистрированы в системе на его получение, тогда в них происходит вызов метода onReceive, где интент присутствует в качестве одного из параметров.
Итак, первое — широковещательный приемник должен быть зарегистрирован для получения конкретных интентов. Во-вторых, какой-либо компонент генерирует интент и передает его в систему. В-третьих, Android доставляет этот интент получателям широковещательных сообщений, которые зарегистрированы для его получения. И четвертое — в приемниках затем происходит вызов их метода onReceive, в котором они и обрабатывают входящее событие.
Теперь поговорим о каждом из этих шагов по порядку. Для регистрации широковещательного приемника у разработчиков есть два варианта. Первый, они могут статически зарегистрировать широковещательный приемник, помещая информацию о нем в файл AndroidManifest. xml приложения, которому принадлежит широковещательный приемник. И второй, они могут регистрировать широковещательный приемник динамически, вызывая определенные методы во время выполнения программы.
Чтобы зарегистрировать широковещательный приемник статически, необходимо добавить в манифест тег «receiver», и затем внутри этого тега необходимо поместить по крайней мере один тег интент-фильтра. По содержимому интент-фильтра Android и определяет, соответствует ли полученный интент этому широковещательному приемнику. Формат тега выглядит примерно так.
Вначале стоит ключевое слово receiver, а затем добавляются некоторые из следующих атрибутов.
android: enabled позволяет включать или отключать определенный приемник.
android: exported, если установлено значение true, этот приемник может принимать широковещательные передачи от других приложений, в то время как если он установлен в значение false, то только те интенты, которые передаются другими компонентами внутри приложения.
android: name дает имя класса, реализующего этот приемник.
android: permission определяет строку разрешения, которое отправитель интента должен иметь, чтобы этот приемник получил интент от них. Как уже говорилось, необходимо указать хотя бы один тег интент-фильтра, которые были рассмотрены в главе Permissions. Теги этого интент-фильтра так же могут указывать на действия (action), данные (data) и категории (categories).
Если регистрировать приемник статически, эта информация будет считана и обработана при установке приложения и во время каждой загрузки системы. Рассмотрим приложение, которое статически регистрирует один широковещательный приемник, который получает пользовательский интент, называемый show toast intent. Это приложение отображает одну кнопку с надписью «Broadcast Intent». Нажатие этой кнопки вызывает отправку интента, а затем его прием широковещательным приемником, который выводит на экран тост-сообщение.
Теперь откроем основную Activity. Этот код сначала определяет строку — действие интента, которая будет использоваться для идентификации этого интента.
Далее идет «слушатель» кнопки, который вызывает метод sendBroadcast, передавая в интенте строку разрешения.
Этот интент будет сопоставлен с зарегистрированными в системе интент-фильтрами. Строка разрешения custom_intent указывает, что этот интент может быть доставлен только тем приемникам широковещательной передачи, у которых есть конкретно это разрешение.
Вы также можете программно регистрировать широковещательные приемники во время выполнения программы, например, если хотите, чтобы ваш широковещательный приемник реагировал только тогда, когда Activity находится на переднем плане. Тогда вы можете динамически регистрировать и отменять регистрацию приемников непосредственно в Activity, например в методах onResume и onPause.
Для этого сначала необходимо создать объект интент-фильтра, указав интенты, которые вы хотите зарегистрировать. Затем необходимо создать широковещательный приемник и зарегистрировать его, вызвав метод registerReceiver. У этого метода есть различные реализации. Одна из них находится в классе LocalBroadcastManager и используется для трансляций внутри одного приложения. Другая реализация находится в классе Context и используется для трансляции интентов по всей системе. Таким образом, эти интенты могут быть получены любым приложением на устройстве. Затем, в случае необходимости, можно вызвать метод unregisterReceiver, чтобы отменить регистрацию этих широковещательных приемников.
Рассмотрим исходный код приложения с динамической регистрацией. Это приложение выглядит так же, как и предыдущее приложение. Есть кнопка «Broadcast Intent» и при её нажатии, транслируется интент. Широковещательный приемник получает этот интент, а затем отображает на экране тост-сообщение, но внутренне его реализация несколько отличается.
Этот код начинается с создания строки действия интента, а затем создается интент-фильтр с только что созданной строкой действия. После этого создается экземпляр широковещательного приемника, называемый receiver. А затем в методе onCreate код получает экземпляр менеджера LocalBroadcastManager, который используется для трансляции и получения интентов в рамках только этого приложения. Затем код использует этого менеджера, чтобы зарегистрировать приемник интента, показывающего тост-сообщение.
И после этого, если нажата кнопка Broadcast Intent, «слушатель» кнопки вызывает метод sendBroadcast менеджера, передавая интент как параметр.
Когда кнопка нажата, менеджер отправляет интент объекту-приемнику, который работает так же, как в предыдущем приложении, создавая и отображая тост-сообщение.
До сих пор наш пример просто передавал один интент одному приемнику. Но Android поддерживает и множество других способов, с помощью которых можно транслировать интенты. Например, интенты могут транслироваться нормально или по очереди.
Нормальная широковещательная передача доставляется подписанным широковещательным приемникам в неопределенном порядке. Таким образом, если у вас есть два широковещательных приемника, которые должны получить один интент, то возможно, что оба приемника смогут обработать этот интент одновременно.
Упорядоченная широковещательная передача, с другой стороны, доставляет интент нескольким широковещательным приемникам по одному в приоритетном порядке.
Трансляции также могут быть липучими или нелипучими. Липучий интент, после того как был транслирован, «крутится поблизости». Таким образом, широковещательный приемник, который был зарегистрирован позже произошедшей трансляции интента, все равно сможет принять его. Эта функция полезна, например, для записи изменений состояния системы, таких как, изменения уровня заряда батареи или состояния зарядки. В этих случаях для приемников не имеет значения, когда состояние изменилось, а только то, в каком текущем состоянии находится система.
Нелипучие трансляции после их первоначальной трансляции отменяются. Эти трансляции больше подходят для сообщения о том, что произошло определенное событие. В этих случаях, если приемник не был зарегистрирован для приема широковещательной передачи до того как она произошла, тогда он и не получит её.
И, наконец, как видно в примерах, некоторые методы трансляции также принимают строковый параметр — разрешение. Это ограничивает широковещательную рассылку теми приемниками, которые имеют указанное разрешение.
Если возникают проблемы при приеме трансляций правильными приемниками вещания, то есть несколько вещей, которые можно сделать. Использовать логи с дополнительной информацией о разрешениях интентов в процессах Android. Это можно сделать, установив флаг FLAG_DEBUG_LOG_RESOLUTION в интенте, который вы собираетесь транслировать.
Другое — использовать инструмент ADB, чтобы узнать, какие широковещательные приемники были зарегистрированы. Можно использовать команду dumpsys для отображения широковещательных приемников, которые зарегистрированы динамически:
Или чтобы показать те, которые зарегистрированы статически:
Теперь, после того, как интент передан, он будет доставлен соответствующему широковещательному приемнику путем вызова его метода onReceive. Этот метод имеет два параметра. Один — Context, в котором работает приемник, и два — интент, который был передан. Поэтому при написании кода, который обрабатывает входящие трансляции, следует учитывать некоторые вещи.
Во-первых, для доставки широковещательной передачи, возможно, системе придется запустить процесс приложения приемника, потому что во время передачи интента он может не работать. И пока onReceive запущен, этот процесс будет иметь высокий приоритет. Таким образом, вещание интента, особенно если оно направлено нескольким приложениям, может быть относительно ресурсоёмкой операцией.
Во-вторых, метод onReceive запускается в главном потоке процесса, следовательно, он должен работать кратковременно, чтобы не блокировать основной поток. В частности, если обработка, которую необходимо сделать в ответ на широковещательную рассылку, занимает много времени, следует подумать о запуске сервиса (нового потока) для этой работы, вместо того, чтобы выполнять её в onReceive.
Фактически, именно это делает приложение SMS, когда приходит SMS-сообщение. Широковещательный приемник принимает широковещательную рассылку и в своем методе onReceive, он немедленно передает это сообщение другой службе, которая проделает всю работу по загрузке сообщения, сохранению его в базе данных и т. д.
Последний вопрос, который следует учитывать, заключается в том, что широковещательный приемник считается действующим только до тех пор, пока запущен OnReceive. На самом деле, как только произойдет возврат из функции OnReceive, Android завершит работу вызвавшего её приемника. В частности, это означает, что широковещательный приемник не может запускать асинхронные операции, которые необходимо будет выполнить в более позднее время.
Рассмотрим несколько примеров приложений, использующих широковещательные приемники. Запустим приложение Compound Broadcast. На экране имеется кнопка, обозначенная «Broadcast Intent», которая при нажатии будет использовать метод sendBroadcast для трансляции интента ShowToast.
Однако на этот раз есть три широковещательных приемника, которые получат этот интент.
Если теперь нажать эту кнопку, ниже появится тост-сообщение от приемника 1, затем второе от приемника 2 и третье от приемника 3. Это происходит потому, что используемое приложение отправило широковещательную рассылку. Передачи рассылались нормально. Это означает, что порядок поступления и обработки данных не был определен. В этом случае вещание обрабатывалось в одном конкретном порядке. В других случаях или на другом устройстве, или если Android изменит свое состояние в будущем, этот порядок может отличаться.
Поэтому, если необходимо, чтобы широковещательные приемники принимали широковещательные передачи в определенном порядке или если необходимо, чтобы каждый широковещательный приемник имел в процессе эксклюзивный доступ к интенту, тогда придется сделать трансляцию упорядоченной. Это делается при помощи метода sendOrderedBroadcast или используя некоторые из методов класса Context. Первый отправляет интент для приемников, у которых есть указанное разрешение в порядке приоритета.
Второе, делает то же самое, но предоставляет дополнительные параметры для большего контроля.
Рассмотрим приложения, которые используют некоторые варианты метода sendOrderedBroadcasts. Приложение Compound Ordered Broadcast отображает кнопку с надписью «Broadcast Intent», и при нажатии этой кнопки, транслируется интент showToast.
У этого приложения есть три зарегистрированных широковещательных приемника: Receiver 1, Receiver 2 и Receiver 3. Все имеют разный приоритет для приема широковещательной передачи. Второй приемник имеет наивысший приоритет, следующим идет первый приемник, а третий приемник имеет самый низкий приоритет. Поэтому ожидается, что приемники получат широковещательную передачу в таком порядке: приемник два, приемник один, приемник три. Тем не менее, в первый приемник помещен некоторый код, который прерывает трансляцию. Таким образом, только приемник два, а затем приемник один должны получить широковещательную рассылку.
Рассмотрим, как это работает. При нажатии кнопки Broadcast Intent появляется тост-сообщение, что интент был получен приемником два, потом еще одно сообщение, что интент получен приемником один. А до бедного приемника три рассылка не дошла.
Рассмотрим исходный код этого приложения. Начнем с открытия файла manifest. xml. Видно, что Receiver2 статически зарегистрирован с приоритетом 10, а Receiver3 зарегистрирован с приоритетом 1.
Теперь перейдем в основную Activity. Код создает экземпляр Receiver1, затем создает интент-фильтр для интента show_toast, а затем присваивает приоритету значение три. Таким образом, любой экземпляр приемника Receiver2 имеет приоритет 10. Этот экземпляр приемника Receiver1 имеет приоритет 3. А приемник три в качестве приоритета имеет 1.
После того как будет нажата кнопка «Broadcast Intent», «слушатель» вызывает sendOrderedBroadcast, передавая как параметр новый интент show_toast. Этот интент сначала получает экземпляр приемника 2, а затем экземпляр приемника 1.
Далее откроем код приемника 1. В onReceive проверяется, является ли трансляция упорядоченной. И если это так, вызывается abortBroadcast, который уничтожает широковещательную рассылку, что предотвращает ее отправку приемнику 3.
Теперь давайте рассмотрим несколько более сложное использование прерывания трансляции. В этом приложении, называемом OrderedBroadcastWithResultReceiver, отображается одно тост-сообщение, которое показывает все широковещательные приемники, получающие этот интент, в том порядке, в котором они были получены.
Рассмотрим основную Activity, «слушатель» кнопки «Broadcast Intent» использует вторую форму метода send broadcast, в котором нас интересует параметр BroadcastReceiver. Назовем его результирующим приемником. Он получит интент после того, как все другие приемники получили возможность принять его. И это полезно, если исходные широковещательные приемники что-то вычисляют, тогда этот результат будет доступен для этого последнего приемника. И он может получить этот результат, вызвав getResultData.
Теперь давайте рассмотрим один из классов вещательных приемников, чтобы увидеть, как они вычисляют результат, который результирующий приемник отображает в тост-сообщении. Это класс Receiver1, его метод onReceive вызывает getResultData, чтобы получить текущие результирующие данные. Затем он добавляет в конец строки свои данные и сохраняет новую строку, вызывая setResultData.
Поскольку все широковещательные приемники по очереди добавляют свои данные в конец строки, то окончательный результат представляют собой одну строку, показывающую, какие приемники получили интент и в каком порядке.
Последний вид трансляции — это липучие трансляции. Как ранее отмечалось, нелипучие трансляции показывают, что что-то произошло в определенный момент времени. Они объявляют о событиях. И, как только событие закончилось, о нем забыли. Android ликвидирует это событие и идет дальше. Если вещательный приемник не был зарегистрирован для получения соответствующего интента, когда произошло его событие, например вчера, тогда об этом уже поздно говорить сегодня.
Однако бывают и другие события, указывающие на изменения состояния, которые могут сохраняться с течением времени. Например, широковещательные приемники, которые хотят знать о текущем состоянии устройства, даже если они не были зарегистрированы в тот момент, когда событие произошло, все равно должны быть уведомлены о том, когда это конкретное состояние изменилось. Например, если уровень заряда аккумулятора очень низкий, Android будет транслировать этот факт в систему. Уважающее себя приложение не станет выполнять операции, потребляющие много энергии батареи, когда устройство находится в состоянии низкого заряда. Поэтому, если новое приложение запускается и регистрируется, то чтобы узнать состояние батареи, этому приложению нужно получить информацию прямо сейчас, находится ли устройство в состоянии разряженной батареи. Вот для этого Android транслирует липучий интент о низком заряде батареи.
Android кэширует липучие интенты, а липучие трансляции данного интента перезаписывают предыдущие кэшированные значения соответствующего интента. Когда широковещательный приемник зарегистрируется динамически, ему будут транслировны все кэшированные интенты, соответствующие его интент-фильтру. Далее, подобно нелипучим передачам, липучие трансляции могут быть отправлены нормальным образом, то есть без определенного порядка или упорядоченно в порядке приоритета. И, наконец, приложения, которые хотят транслировать липучие интенты, должны иметь разрешение BROADCAST_STICKY.
Рассмотрим пример приложения, использующего липучие трансляции. Представим приложение Sticky intent, открытое в эмуляторе, которым можно управлять при помощи терминала telnet. Приложение отображает текущий уровень заряда батареи, а также отображает строку «Reading may be stale», которая указывает, что считанный уровень заряда батареи взят из кэшированной липучей трансляции, и может оказаться устаревшим. Если в терминале изменить уровень заряда батареи, то будет транслирован новый интент, отражающий обновленный уровень заряда батареи. Будут обновлены показания дисплея, и изменится текст, чтобы показать, что широковещательный приемник может различать свежие трансляции и кэшированные.
Далее перейдем к исходному коду этого приложения, к основной Activity. Здесь происходит регистрация нового широковещательного приемника.
Бесплатный фрагмент закончился.
Купите книгу, чтобы продолжить чтение.