1. 前言
Service意为:服务,是一种运行时用户不可见的活动机制。可以理解为它是一个没有布局的Activity。
典型的场景:音乐后台播放、后台下载。
Service不同于子线程,Service是运行在主线程中的,因此不能进行耗时操作。
2. 代码编写
使用android studio创建一个空项目(empty activity)
创建完成后就是一个可以显示"hello world"的小应用了。
点击MainActivity.java里的“activity_main”进入activity_main.xml,在这里可以点击“design”,创建button。
我创建button添加的xml代码如下:
<Button
android:id="@+id/button"
android:layout_width="314dp"
android:layout_height="64dp"
android:text="@string/startservice"
android:onClick="startService"
tools:layout_editor_absoluteX="52dp"
tools:layout_editor_absoluteY="70dp"
tools:ignore="MissingConstraints" />
“android:text= ”后是按钮描述
“android:onClick”后是按钮点击事件,可以右击生成点击方法,选择点击方法位于MainActivity.java
点击方法是开启service的入口点
生成的方法是
public void startService(View view) {
}
注意,在修改button的参数值时,res->values->strings.xml可能也会自动生成新的内容
<resources>
<string name="app_name">Zyy Application</string>
<string name="hello_my_world">Hello my World!</string>
<string name="startservice">启动Service</string>
</resources>
接下来新建service,取名为 MyService。
新建完后用“ctrl + o” 选择重写方法。并设计自定义的service执行内容。我这里编写的是每隔一秒打印时间,一共打印五次。
package com.example.zyyapplication;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.util.Log;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
public class MyService extends Service {
private static final String TAG = "Service";
public MyService() {
}
@Override
public IBinder onBind(Intent intent) {
Log.d(TAG, "onBind: ");
// TODO: Return the communication channel to the service.
throw new UnsupportedOperationException("Not yet implemented");
}
@Override
public void onCreate() {
super.onCreate();
Log.d(TAG, "onCreate: ");
}
@Override
public void onDestroy() {
super.onDestroy();
Log.d(TAG, "onDestroy: ");
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.d(TAG, "onStartCommand: ");
int i = 0;
String[] arr = {"date"};
for (i = 1; i < 5; i++) {
System.out.println("i is:" + i);
try {
Thread.sleep(1000);
String result = execcommand(arr);
System.out.println("exec command result is:" + result);
} catch (IOException | InterruptedException e) {
e.printStackTrace();
}
}
return super.onStartCommand(intent, flags, startId);
}
public static String execcommand(String[] command) throws IOException {
BufferedReader bufferedReader = null;
try {
Process process = Runtime.getRuntime().exec(command);
bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()));
final String s = bufferedReader.readLine();
return s;
} finally {
bufferedReader.close();
}
}
}
在AndroidManifest.xml中注册这个类,填写一行service信息
</activity>
<service android:name=".MyService" />
</application>
最后,不要忘了在MainActivity.java里对service编写启动代码。
public void startService(View view) {
Intent intent =new Intent(this, MyService.class);
startService(intent);
}
3.调试
选择设备型号
我随便选了个Pixel API 28
接下来就可以调试了
点击运行
出现手机界面,点击界面上的启动service。
日志打印如下:
D/Service: onCreate:
D/Service: onStartCommand:
I/System.out: i is:1
I/System.out: exec command result is:Wed Dec 21 17:59:32 CST 2022
i is:2
I/System.out: exec command result is:Wed Dec 21 17:59:33 CST 2022
i is:3
I/System.out: exec command result is:Wed Dec 21 17:59:34 CST 2022
i is:4
I/System.out: exec command result is:Wed Dec 21 17:59:35 CST 2022
I/Choreographer: Skipped 241 frames! The application may be doing too much work on its main thread.
I/OpenGLRenderer: Davey! duration=4039ms; Flags=0, IntendedVsync=8909016702842, Vsync=8913033369348, OldestInputEvent=9223372036854775807, NewestInputEvent=0, HandleInputStart=8913046014660, AnimationStart=8913046074360, PerformTraversalsStart=8913046841460, DrawStart=8913047184160, SyncQueued=8913048173260, SyncStart=8913048223960, IssueDrawCommandsStart=8913048276060, SwapBuffers=8913054677760, FrameCompleted=8913056277560, DequeueBufferDuration=154000, QueueBufferDuration=341000,
从日志中我们可以看出,service正常启动了。
4.总结
有一个视频,讲的比较通俗易懂。推荐给大家。我写得比较简略,如果有不明白得地方可以搜索 Bi站视频 “Android中Service的介绍”,博主 “子林android”。