Github: https://github.com/MADMAX110/Joke
服务是与活动类似的应用组件,只不过服务没有用户界面。
使用服务可以一直在后台做某些事情,比如下载一个大文件,播放一段音乐或者监听来自服务器的一个消息。
有三种类型的服务:
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");
}
@Override
protected 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");
}
@Override
protected 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">
<application
android: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">
<activity
android: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">
<Button
android: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 {
@Override
protected 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,//不需要访问挂起意图时,可以将其设置为0
actionIintent,//这是上面创建的意图
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");
}
@Override
protected 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,//不需要访问挂起意图时,可以将其设置为0
actionIintent,//这是上面创建的意图
PendingIntent.FLAG_UPDATE_CURRENT);//如果有一个匹配的挂起意图,它会更新
//这会把这个挂起意图增加到通知
builder.setContentIntent(actionPendingIntent);
//允许你访问Android的通知服务
NotificationManager notificationManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//使用通知服务来显示我们创建的通知
notificationManager.notify(NOTIFICATION_ID, builder.build());
}
}