Сервисы представляют собой особую организацию приложения. В отличие от activity они не требуют наличия визуального интерфейса. Сервисы позволяют выполнять долговременные задачи без вмешательства пользователя.
Все сервисы наследуются от класса Service и проходят следующие этапы жизненного цикла:
Метод onCreate()
: вызывается при создании сервиса
Метод onStartCommand()
: вызывается при получении сервисом команды, отправленной с помощью метода startService()
Метод onBind()
: вызывается при закреплении клиента за сервисом с помощью метода bindService()
Метод onDestroy()
: вызывается при завершении работы сервиса
Создадим простейшее приложение с сервисом. Наш сервис будет воспроизводить музыкальный файл. И вначале добавим в проект в каталог res папку raw. Для этого нажмем правой кнопкой мыши на каталог res и в контекстном меню выберем пункт New -> Android Resource Directory.
Далее укажем в качестве типа папки - raw:
И поместим в эту папку (res/raw) какой-нибудь mp3-файл.
Затем добавим новый класс сервиса. Назовем его MediaService. В итоге получится следующий проект:
Для воспроизведения аудио-файла определим в классе MediaService следующий код:
package com.example.soundserviceapp; import android.app.Service; import android.content.Intent; import android.media.MediaPlayer; import android.os.IBinder; public class MediaService extends Service { MediaPlayer ambientMediaPlayer; @Override public IBinder onBind(Intent intent) { throw new UnsupportedOperationException("Not yet implemented"); } @Override public void onCreate(){ ambientMediaPlayer=MediaPlayer.create(this, R.raw.music); ambientMediaPlayer.setLooping(true); } @Override public int onStartCommand(Intent intent, int flags, int startId){ ambientMediaPlayer.start(); return START_STICKY; } @Override public void onDestroy() { ambientMediaPlayer.stop(); } }
Для воспроизведения музыкального файла сервис будет использовать компонент MediaPlayer.
В сервисе переопределяются все четыре метода жизненного цикла. Но по сути метод onBind()
не имеет никакой реализации.
В методе onCreate()
инициализируется медиа-проигрыватель с помощью музыкального ресурса, который добавлен в папку res/raw.
В методе onStartCommand()
начинается воспроизведение.
Метод onStartCommand()
может возвращать одно из значений, которое предполагает различное поведение в случае, если процесс сервиса был неожиданно
завершен системой:
START_STICKY: в этом случае сервис снова возвращается в запущенное состояние, как будто если бы снова был бы вызван метод onStartCommand() без передачи в этот метод объекта Intent
START_REDELIVER_INTENT: в этом случае сервис снова возвращается в запущенное состояние, как будто если бы снова был бы вызван метод onStartCommand() с передачей в этот метод объекта Intent
START_NOT_STICKY: сервис остается в остановленном положении
Метод onDestroy()
завершает воспроизведение.
Чтобы управлять сервисом, изменим activity. Сначала добавим в файл activity_main.xml пару кнопок для управления сервисом:
<?xml version="1.0" encoding="utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent"> <Button android:id="@+id/start" android:layout_width="0dp" android:layout_height="wrap_content" android:text="Старт" android:onClick="click" app:layout_constraintLeft_toLeftOf="parent" app:layout_constraintRight_toLeftOf="@id/stop" app:layout_constraintTop_toTopOf="parent" /> <Button android:id="@+id/stop" android:layout_width="0dp" android:layout_height="wrap_content" android:text="Стоп" android:onClick="click" app:layout_constraintLeft_toRightOf="@id/start" app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toTopOf="parent" /> </androidx.constraintlayout.widget.ConstraintLayout>
И изменим код MainActivity:
package com.example.soundserviceapp; import androidx.appcompat.app.AppCompatActivity; import android.content.Intent; import android.os.Bundle; import android.view.View; public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } public void click(View v) { Intent i=new Intent(this, MediaService.class); if (v.getId()==R.id.start) { startService(i); } else { stopService(i); } } }
Для запуска сервиса используется объект Intent:
Intent i=new Intent(this, MediaService.class);
Для запуска сервиса в классе Activity определен метод startService(), в который передается объект Intent. Этот метод
будет посылать команду сервису и вызывать его метод onStartCommand()
, а также указывать системе, что сервис должен продолжать работать
до тех пор, пока не будет вызван метод stopService().
Метод stopService() также определен к классе Activity и принимает объект Intent. Он останавливает работу сервиса, вызывая
его метод onDestroy()
И в конце нам надо зарегистрировать сервис в файле манифеста:
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.soundserviceapp"> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/Theme.SoundServiceApp"> <service android:name=".MediaService" android:enabled="true" android:exported="true" > </service> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> </manifest>
Регистрация сервиса производится в узле application
с помощью добавления элемента <service>
. В нем определяется атрибут
android:name
, который хранит название класса сервиса. И кроме того может принимать еще ряд атрибутов:
android:enabled
: если имеет значение "true", то сервис может ли создаваться системой. Значение по умолчанию - "true".
android:exported
: указывает, могут ли компоненты других приложений обращаться к сервису. Если имеет значение "true", то могут,
если имеет значение "false", то нет.
android:icon
: значок сервиса, представляет собой ссылку на ресурс drawable
android:isolatedProcess
: если имеет значение true, то сервис может быть запущен как специальный процесс, изолированный от остальной системы.
android:label
: название сервиса, которое отображается пользователю
android:permission
: набор разрешений, которые должно применять приложение для запуска сервиса
android:process
: название процесса, в котором запущен сервис. Как правило, имеет то же название, что и пакет приложения.
Запустим приложение и нажмем на кнопку запуска сервиса:
После этого начнется воспроизведение добавленной нами в приложение мелодии.