ฉันเจอพฤติกรรมในการจัดการกระบวนการของ Android ร่วมกับบริการเบื้องหน้าที่ทำให้ฉันสับสนจริงๆ
มีเหตุผลอะไรสำหรับฉัน
- เมื่อคุณปัดแอพของคุณจาก 'แอพล่าสุด' ระบบปฏิบัติการควรเสร็จสิ้นกระบวนการแอปในอนาคตอันใกล้
- เมื่อคุณปัดแอพของคุณจาก 'แอพล่าสุด' ขณะที่ใช้บริการเบื้องหน้าแอพจะยังคงอยู่
- เมื่อคุณหยุดให้บริการเบื้องหน้าก่อนที่จะปัดแอปของคุณจาก 'แอพล่าสุด' คุณจะได้รับเหมือนกับ 1)
ฉันสับสนอะไร
เมื่อคุณหยุดใช้บริการเบื้องหน้าในขณะที่ไม่มีกิจกรรมใดในเบื้องหน้า (แอพไม่ปรากฏใน 'แอพล่าสุด') ฉันคาดว่าแอพจะถูกฆ่าทันที
อย่างไรก็ตามสิ่งนี้ไม่ได้เกิดขึ้นกระบวนการแอพยังมีชีวิตอยู่
ตัวอย่าง
ฉันได้สร้างตัวอย่างขั้นต่ำที่แสดงพฤติกรรมนี้
ForegroundService:
import android.app.Notification
import android.app.NotificationChannel
import android.app.NotificationManager
import android.app.PendingIntent
import android.app.Service
import android.content.Context
import android.content.Intent
import android.os.Build
import android.os.IBinder
import androidx.core.app.NotificationCompat
import timber.log.Timber
class MyService : Service() {
override fun onBind(intent: Intent?): IBinder? = null
override fun onCreate() {
super.onCreate()
Timber.d("onCreate")
}
override fun onDestroy() {
super.onDestroy()
Timber.d("onDestroy")
// just to make sure the service really stops
stopForeground(true)
stopSelf()
}
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
Timber.d("onStartCommand")
startForeground(ID, serviceNotification())
return START_NOT_STICKY
}
private fun serviceNotification(): Notification {
createChannel()
val stopServiceIntent = PendingIntent.getBroadcast(
this,
0,
Intent(this, StopServiceReceiver::class.java),
PendingIntent.FLAG_UPDATE_CURRENT
)
return NotificationCompat.Builder(this, CHANNEL_ID)
.setSmallIcon(R.drawable.ic_launcher_foreground)
.setContentTitle("This is my service")
.setContentText("It runs as a foreground service.")
.addAction(0, "Stop", stopServiceIntent)
.build()
}
private fun createChannel() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
val notificationManager = getSystemService(NotificationManager::class.java)
notificationManager.createNotificationChannel(
NotificationChannel(
CHANNEL_ID,
"Test channel",
NotificationManager.IMPORTANCE_DEFAULT
)
)
}
}
companion object {
private const val ID = 532207
private const val CHANNEL_ID = "test_channel"
fun newIntent(context: Context) = Intent(context, MyService::class.java)
}
}
BroadcastReceiver เพื่อหยุดบริการ:
import android.content.BroadcastReceiver
import android.content.Context
import android.content.Intent
class StopServiceReceiver : BroadcastReceiver() {
override fun onReceive(context: Context, intent: Intent) {
val serviceIntent = MyService.newIntent(context)
context.stopService(serviceIntent)
}
}
กิจกรรม:
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
startService(MyService.newIntent(this))
}
}
รายการ:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
package="com.christophlutz.processlifecycletest">
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<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/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<service android:name=".MyService"/>
<receiver android:name=".StopServiceReceiver" />
</application>
</manifest>
ลองใช้วิธีต่อไปนี้:
- เริ่มแอพหยุดบริการเบื้องหน้านำแอพออกจาก 'แอพล่าสุด'
- เริ่มแอพลบแอพจาก 'แอพล่าสุด' หยุดบริการส่วนหน้า
คุณสามารถดูได้ใน LogCat ของ Android Studio ว่ากระบวนการแอปมีการทำเครื่องหมาย [DEAD] สำหรับกรณีที่ 1 แต่ไม่ใช่สำหรับกรณีที่ 2
เนื่องจากมันค่อนข้างง่ายที่จะทำซ้ำมันอาจเป็นพฤติกรรมที่ตั้งใจ แต่ฉันไม่พบการพูดถึงสิ่งนี้ในเอกสาร
ไม่มีใครรู้ว่าเกิดอะไรขึ้นที่นี่
onDestroy
(บริการ) ได้รับเรียก แต่กระบวนการยังคงมีชีวิตอยู่นานหลังจากทุกอย่างหายไป แม้ว่าระบบปฏิบัติการจะยังคงใช้งานได้ในกรณีที่บริการเริ่มต้นใหม่ฉันไม่เห็นว่าทำไมมันไม่ทำสิ่งเดียวกันเมื่อคุณหยุดให้บริการก่อนจากนั้นจึงลบแอปออกจากสำรับ พฤติกรรมที่ปรากฏดูเหมือนไม่ได้ใช้งานง่ายโดยเฉพาะอย่างยิ่งการเปลี่ยนแปลงล่าสุดของข้อ จำกัด การดำเนินการพื้นหลังดังนั้นจึงเป็นเรื่องดีที่ได้ทราบวิธีการตรวจสอบให้แน่ใจว่ากระบวนการหยุดทำงาน