服务是与活动类似的应用组件,只不过服务没有用户界面。
使用服务可以一直在后台做某些事情,比如下载一个大文件,播放一段音乐或者监听来自服务器的一个消息。
有三种类型的服务:
1、启动式服务
启动式服务可以在后台无限期的运行,即使启动这个服务的活动已经撤销,也不会影响服务的运行。
2、绑定式服务
绑定式服务会绑定到另一个应用组件,如一个活动。
这个活动可以与绑定式服务交互,发送请求并得到结果。只要与之绑定的组件还在运行,绑定式服务就会一直运行下去。这个组件不再与之绑定时服务将会被撤销。
3、调度式服务
调度式服务是安排在某个特定的事件运行的服务。
创建一个启动式服务
要创建一个新工程,其中包含一个名为MainActivity的活动,以及一个名为DelayedMessageService的服务。只要MainActivity调用DelayedMessageService,它会等待10s,然后显示一段文本。
如图创建一个名为Joke的新工程。
使用IntentService类创建一个基本启动式服务
要创建启动式服务,最简单的方法式扩展IntentService类,因为它提供了你需要的大多数功能。
首先利用一个意图启动这个服务,它会在一个单独的线程中运行指定的代码。
在com.hafd.joke包中创建一个名为DelayedMessageService的Java类。
package com.hafd.joke;import android.app.IntentService;
import android.content.Intent;
import androidx.annotation.Nullable;public class DelayedMessageService extends IntentService {public DelayedMessageService() {super("DelayedMessageService");}@Overrideprotected void onHandleIntent(@Nullable Intent intent) {}
}
记录消息日志
在日志中增加消息非常有用,可以利用日志检查你的代码是否能像预期那样工作。
在Java代码中告诉Android要记录什么,应用运行时可以检查Android日志中的输出。
可以使用Android.util.Log类的以下方法记录消息日志:
Log.v(TAG, "Verbose message"); // 记录详细信息Log.d(TAG, "Debug message"); // 记录调试信息Log.i(TAG, "Info message"); // 记录普通信息Log.w(TAG, "Warning message"); // 记录警告信息Log.e(TAG, "Error message"); // 记录错误信息
完整的DelayedMessageSeervice代码
package com.hafd.joke;import android.app.IntentService;
import android.content.Intent;
import android.util.Log;import androidx.annotation.Nullable;public class DelayedMessageService extends IntentService {public static final String EXTRA_MESSAGE = "message";public DelayedMessageService() {super("DelayedMessageService");}@Overrideprotected void onHandleIntent(@Nullable Intent intent) {synchronized (this) {try {wait(10000);} catch (InterruptedException e) {e.printStackTrace();}}String text = intent.getStringExtra(EXTRA_MESSAGE);showText(text);}private void showText(final String text){Log.v("DelayedMessageService", "The message is: " + text);}
}
在AndroidManifest.xml中声明服务
其中exported表示是否可以在其他应用中使用。
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"><applicationandroid:allowBackup="true"android:dataExtractionRules="@xml/data_extraction_rules"android:fullBackupContent="@xml/backup_rules"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:roundIcon="@mipmap/ic_launcher_round"android:supportsRtl="true"android:theme="@style/Theme.Joke"tools:targetApi="31"><activityandroid:name=".MainActivity"android:exported="true"><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity><service android:name=".DelayedMessageService"android:exported="false"></service></application></manifest>
为activity_main.xml增加一个按钮
增加两个字符串值
<string name="question">What is the secret of comedy?</string><string name="response">Timing!</string>
接下来更新activity_main.xml,让其显示一个按钮:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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:orientation="vertical"android:padding="16dp"tools:context=".MainActivity"><Buttonandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center_horizontal"android:text="@string/question"android:id="@+id/button"android:onClick="onCLick"/></LinearLayout>
MainActivity使用startService启动服务
只要用户单击按钮,就要使用MainActivity的onClick方法启动DelayedeMessageService。从活动启动另一个服务与启动一个活动很类似。先要创建一个显示意图,指定你想要启动的服务。然后使用活动中的startService方法启动这个服务。
package com.hafd.joke;import androidx.appcompat.app.AppCompatActivity;import android.content.Intent;
import android.os.Bundle;
import android.view.View;public class MainActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}public void onClick(View view){Intent intent = new Intent(this, DelayedMessageService.class);intent.putExtra(DelayedMessageService.EXTRA_MESSAGE, getResources().getString(R.string.response));startService(intent);}
}
试一试应用
按下按钮等待10s后切换回AS的logcat页面,会发现如下打印:
2023-10-09 09:42:46.602 8237-8306 DelayedMessageService com.hafd.joke V The message is: Timing!
启动式服务生命周期
在Android中,启动式服务的生命周期可以包括以下四个步骤:
onCreate(): 当服务第一次被创建时执行。这个方法可以在服务创建时进行一些初始化操作。
onStartCommand(): 当服务被启动的时候执行。使用startService()这个方法开启服务,但注意,如果服务已经启动过,那么onCreate()方法不会再被调用。此时服务开始执行,可以长期在后台运行。
onHandleIntent():要把希望服务在后台运行的所有代码增加到onHandleIntent()。
onDestroy(): 当服务不再被使用或被销毁时执行。
一般来说,启动式服务的生命周期就是从onCreate()到onDestroy()。不过需要注意的是,如果启动式服务没有被明确停止,那么它可能会在后台持续运行,即使用户退出了应用或设备重新启动,除非服务被系统销毁或明确停止。
Android内置通知服务
我们要修改Joke应用,在一个通知中显示消息。通知时在应用用户界面之外显示消息。发出一个通知时,它会在状态栏的通知区中显示一个图标。可以在通知抽屉中看到这个通知的详细信息,从屏幕上向下滑动就可以看到通知抽屉:
创建一个通知生成器
首先要做的是创建一个通知生成器。它允许你构建有特定内容和特性的通知。
你创建的每个通知都应当包括一个小图标、一个标题和最简版的一些文本。下面是相应的代码:
NotificationCompat.Builder builder = new NotificationCompat.Builder(this).setSmallIcon(android.R.drawable.sym_def_app_icon) //设置显示小图标.setContentTitle(getString(R.string.question)) //设置标题.setContentText(text) //设置文本.setPriority(NotificationCompat.PRIORITY_HIGH) //设置高优先级.setVibrate(new long[] {0, 1000}) //等待0ms,并震动1000ms.setAutoCancel(true);//用户单击通知时通知会消失
增加一个动作告诉通知单击时要启动哪个活动
//启动MainActivity的普通意图Intent actionIintent = new Intent(this, MainActivity.class);PendingIntent actionPendingIntent = PendingIntent.getActivity(this, //上下文,这里是当前服务0,//不需要访问挂起意图时,可以将其设置为0actionIintent,//这是上面创建的意图PendingIntent.FLAG_UPDATE_CURRENT);//如果有一个匹配的挂起意图,它会更新//这会把这个挂起意图增加到通知builder.setContentIntent(actionPendingIntent);
使用内置通知服务发出通知
//允许你访问Android的通知服务NotificationManager notificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);//使用通知服务来显示我们创建的通知notificationManager.notify(NOTIFICATION_ID, builder.build());
完整的DelayedMessageService.java
package com.hafd.joke;import android.app.IntentService;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Intent;
import android.util.Log;import androidx.annotation.Nullable;
import androidx.core.app.NotificationCompat;public class DelayedMessageService extends IntentService {public static final String EXTRA_MESSAGE = "message";public static final int NOTIFICATION_ID = 5453;public DelayedMessageService() {super("DelayedMessageService");}@Overrideprotected void onHandleIntent(@Nullable Intent intent) {synchronized (this) {try {wait(10000);} catch (InterruptedException e) {e.printStackTrace();}}String text = intent.getStringExtra(EXTRA_MESSAGE);showText(text);}private void showText(final String text){NotificationCompat.Builder builder = new NotificationCompat.Builder(this).setSmallIcon(android.R.drawable.sym_def_app_icon) //设置显示小图标.setContentTitle(getString(R.string.question)) //设置标题.setContentText(text) //设置文本.setPriority(NotificationCompat.PRIORITY_HIGH) //设置高优先级.setVibrate(new long[] {0, 1000}) //等待0ms,并震动1000ms.setAutoCancel(true);//用户单击通知时通知会消失//启动MainActivity的普通意图Intent actionIintent = new Intent(this, MainActivity.class);PendingIntent actionPendingIntent = PendingIntent.getActivity(this, //上下文,这里是当前服务0,//不需要访问挂起意图时,可以将其设置为0actionIintent,//这是上面创建的意图PendingIntent.FLAG_UPDATE_CURRENT);//如果有一个匹配的挂起意图,它会更新//这会把这个挂起意图增加到通知builder.setContentIntent(actionPendingIntent);//允许你访问Android的通知服务NotificationManager notificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);//使用通知服务来显示我们创建的通知notificationManager.notify(NOTIFICATION_ID, builder.build());}
}