Files
Obsidian-Main/03. 專注Study/Android/Service.md
Awin Huang 5e4265b783 vault backup: 2022-09-30 21:53:03
Affected files:
.obsidian/workspace
03. 專注Study/Android/ADB 取得 APK 的 icon.md
03. 專注Study/Android/ADB.md
03. 專注Study/Android/AOSP.md
03. 專注Study/Android/Android programming.md
03. 專注Study/Android/AudioTrack.md
03. 專注Study/Android/Ktor.md
03. 專注Study/Android/Service.md
03. 專注Study/Android/Tools.md
03. 專注Study/Android/UI.md
03. 專注Study/C++/C++17.md
03. 專注Study/C++/Class template.md
03. 專注Study/C++/GCC.md
03. 專注Study/C++/Structured binding declaration.md
03. 專注Study/C++/for_each.md
03. 專注Study/C++/lambda.md
03. 專注Study/C++/lvalue.md
03. 專注Study/C++/move operator.md
03. 專注Study/C++/rvalue.md
03. 專注Study/C++/智慧指標.md
03. 專注Study/RxKotlin/20200207 - Study RxKotlin.md
04. Programming/COM/20210726 - COM Interface.md
04. Programming/DB/MySQL.md
04. Programming/DB/sqlite.md
04. Programming/Design Pattern.md
04. Programming/FFMPEG/00. Introduction.md
04. Programming/FFMPEG/01. Setup.md
04. Programming/FFMPEG/FFMpeg.md
04. Programming/Flask.md
04. Programming/Kotlin/class.md
04. Programming/Kotlin/run, let, with, also 和 apply.md
04. Programming/Media Foundation/20210604 - Windows media foundation.md
04. Programming/OpenCV.md
04. Programming/OpenGL.md
04. Programming/Python/argparse.ArgumentParser.md
04. Programming/Python/decorator.md
04. Programming/Python/logging.md
04. Programming/Python/opencv.md
04. Programming/Python/subprocess.md
04. Programming/Python/threading.md
04. Programming/Python/tkinter.md
04. Programming/Python/檢測工具.md
04. Programming/QT/Dropdown button.md
04. Programming/QT/QVariant.md
04. Programming/QT/Qt.md
04. Programming/UML.md
04. Programming/演算法.md
05. 資料收集/99. templates/blogHeader.md
05. 資料收集/99. templates/date.md
05. 資料收集/99. templates/front matter.md
05. 資料收集/99. templates/note.md
05. 資料收集/99. templates/table.md
05. 資料收集/99. templates/thisWeek.md
05. 資料收集/99. templates/日記.md
05. 資料收集/99. templates/讀書筆記.md
05. 資料收集/Linux/CLI/cut.md
05. 資料收集/Linux/CLI/scp.md
05. 資料收集/Linux/CLI/timedatectl.md
05. 資料收集/Linux/Programming.md
05. 資料收集/Linux/Ubuntu.md
05. 資料收集/Tool Setup/Hardware/RaspberryPi.md
05. 資料收集/Tool Setup/Software/Chrome.md
05. 資料收集/Tool Setup/Software/Obisidian.md
05. 資料收集/Tool Setup/Software/SublimeText.md
05. 資料收集/Tool Setup/Software/VirtualBox.md
05. 資料收集/Tool Setup/Software/Visual Studio Code.md
05. 資料收集/Tool Setup/Software/Windows Setup.md
05. 資料收集/Tool Setup/Software/Windows Terminal.md
05. 資料收集/Tool Setup/Software/freefilesync.md
05. 資料收集/Tool Setup/Software/vim.md
05. 資料收集/名言佳句.md
05. 資料收集/架站/Gitea.md
05. 資料收集/架站/HTTP Server/Apache.md
05. 資料收集/架站/HTTP Server/Nginx/Reverse Proxy(Layer4).md
05. 資料收集/架站/Pelican blog.md
05. 資料收集/架站/Proxmox VE.md
05. 資料收集/架站/SWAG Reverse proxy.md
05. 資料收集/架站/Storj.md
05. 資料收集/架站/Trojan.md
05. 資料收集/每週外食.md
05. 資料收集/科技/802.11.md
05. 資料收集/科技/HDR Sensor.md
05. 資料收集/科技/量子電腦.md
05. 資料收集/科技/鋰電池.md
05. 資料收集/興趣嗜好/RC/Traxxas Sledge.md
05. 資料收集/興趣嗜好/RC/好盈電變調整中立點.md
05. 資料收集/興趣嗜好/RC/差速器調教教學.md
05. 資料收集/興趣嗜好/模型/舊化作例.md
05. 資料收集/興趣嗜好/軍武/虎式.md
05. 資料收集/讀書筆記/20201201 - 學習如何學習.md
05. 資料收集/讀書筆記/20201218 - Kotlin權威2.0.md
05. 資料收集/讀書筆記/20201224 - 寫作是最好的自我投資.md
05. 資料收集/讀書筆記/20210119 - 中產悲歌.md
05. 資料收集/讀書筆記/20210220 - 最高學習法.md
05. 資料收集/讀書筆記/20210320 - 最高學以致用法.md
05. 資料收集/讀書筆記/20210406 - 精準購買.md
05. 資料收集/讀書筆記/20210723 - 高手學習.md
05. 資料收集/讀書筆記/20220526 - 深入淺出設計模式.md
05. 資料收集/讀書筆記/20220619 - 精確的力量.md
05. 資料收集/軟體工具/IPFS.md
05. 資料收集/軟體工具/MkDocs.md
05. 資料收集/軟體工具/Obsidian.md
05. 資料收集/軟體工具/docker.md
05. 資料收集/軟體工具/git/apply.md
05. 資料收集/軟體工具/git/submodule.md
05. 資料收集/軟體工具/youtube-dl.md
05. 資料收集/面試準備/技术面试最后反问面试官的话.md
2022-09-30 21:53:04 +08:00

9.1 KiB
Raw Blame History

Service的生命週期

!Pasted image 20220307103552.png

Service的啟動方式

Service由startService()啟動之後便獨立動作啟動者例如某個activity無法取得Service的intance也無法呼叫Service的API。Service可以被多次呼叫startService(),但是只要一旦stopService()被呼叫了Service就會結束。所以需要確保Service的管理者是誰統一管理者來呼叫startService()stopService()才不會造成混亂。

bindService()則是像是典型的Server-Client架構第一次bindService()的時候會建立Service的instance然後可以多次bindService()unbindService()當Service沒也任何人跟它"Bind"的時候Service才會結束。

startService

要建立自己的Service需要繼承Service()類別然後複寫4個成員函式

class MyService : Service() {

    override fun onCreate() {
        Log.i("Awin","onCreate - Thread ID = " + Thread.currentThread().id)
        super.onCreate()
    }

    override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
        Log.i("Awin", "onStartCommand - startId = " + startId + ", Thread ID = " + Thread.currentThread().id)
        return super.onStartCommand(intent, flags, startId)
    }

    override fun onBind(p0: Intent?): IBinder? {
        Log.i("Awin", "onBind - Thread ID = " + Thread.currentThread().id)
        return null  // 因為目前沒有要支援bindService()所以這裡直接return null
    }

    override fun onDestroy() {
        Log.i("Awin", "onDestroy - Thread ID = " + Thread.currentThread().id)
        super.onDestroy()
    }
}

假設現在由MainActivity呼叫startService()因為是第一次呼叫所以MyService的callback被呼叫的順序是onCreate() -> onStartCommand()。 第二次之後的呼叫就只會執行onStartCommand()。 當MainActivity呼叫stopService()則會執行MyService的onDestroy()

所以,必須將「長期任務」開個 Thread 並且執行在 onStartCommand() 方法內。

Client端的code:

val intent = Intent(this, ServiceDemo.class)
startService(intent)

bindService

使用bindService()來建立Service的一個好處就是可以取得Service的instance然後藉由這個instance來使用Service的API。 要使用bindService()的話Service類別必須實做onBind()onUnbind()至兩個override function。 也必須提供一個Binder class繼承自Binder()來讓client呼叫。例

class MyService : Service() {

    //client 可以通过Binder获取Service实例
    inner class MyBinder : Binder() {
        val service: MyService
        get() = this@MyService
    }

    //通过binder实现调用者client与Service之间的通信
    private val binder = MyBinder()

    private val generator: Random = Random()

    override fun onCreate() {
        Log.i("xiao", "MyService - onCreate - Thread = " + Thread.currentThread().name)
        super.onCreate()
    }

    /**
     *  @param intent 啟動時啟動組件傳遞過來的Intent如Activity可利用Intent封裝所需要的參數並傳遞給Service
     *  @param flags 表示啟動請求時是否有額外數據,可選值有 0START_FLAG_REDELIVERYSTART_FLAG_RETRY
     *  0: 在正常創建Service的情況下onStartCommand傳入的flags為0。
     *
     *  START_FLAG_REDELIVERY:
     *  這個值代表了onStartCommand()方法的返回值為 START_REDELIVER_INTENT
     *  而且在上一次服務被殺死前會去調用stopSelf()方法停止服務。
     *  其中START_REDELIVER_INTENT意味著當Service因記憶體不足而被系統kill後
     *  則會重建服務,並透過傳遞給服務的最後一個 Intent調用 onStartCommand()此時Intent時有值的。
     *
     *  START_FLAG_RETRY
     *  該flag代表當onStartCommand()調用後一直沒有返回值時會嘗試重新去調用onStartCommand()。
     *
     *  @param startId 指明當前服務的唯一ID與stopSelfResult(int startId)配合使用stopSelfResult()可以更安全地根據ID停止服務。
     *
     *  @return
     *  START_STICKY:
     *  當Service因記憶體不足而被系統kill後一段時間後記憶體再次空閒時
     *  系統將會嘗試重新創建此Service一旦創建成功後將回調onStartCommand方法
     *  但其中的Intent將是null除非有掛起的Intent如pendingintent
     *  這個狀態下比較適用於不執行命令、但無限期運行並等待作業的媒體播放器或類似服務
     *
     *
     *  START_NOT_STICKY:
     *  當Service因記憶體不足而被系統kill後即使系統記憶體再次空閒時
     *  系統也不會嘗試重新創建此Service。除非程序中再次調用startService啟動此Service
     *  這是最安全的選項,可以避免在不必要時以及應用能夠輕鬆重啟所有未完成的作業時運行服務。
     *
     *  START_REDELIVER_INTENT:
     *  當Service因記憶體不足而被系統kill後則會重建服務
     *  並透過傳遞給服務的最後一個 Intent 調用 onStartCommand(),任何掛起 Intent均依次傳遞。
     *  與START_STICKY不同的是其中的傳遞的Intent將是非空是最後一次調用startService中的intent。
     *  這個值適用於主動執行應該立即恢復的作業(例如下載文件)的服務。
     */
    override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
        Log.i("xiao", "MyService - onStartCommand - startId = $startId, Thread = " + Thread.currentThread().name)
        return START_NOT_STICKY
    }

    override fun onBind(intent: Intent): IBinder{
        Log.i("xiao", "MyService - onBind - Thread = " + Thread.currentThread().name)
        return binder
    }

    override fun onUnbind(intent: Intent): Boolean {
        Log.i("xiao", "MyService - onUnbind - from = " + intent.getStringExtra("from"))
        return super.onUnbind(intent)
    }

    override fun onDestroy() {
        Log.i("xiao", "MyService - onDestroy - Thread = " + Thread.currentThread().name)
        super.onDestroy()
    }

    //getRandomNumber是Service暴露出去供client调用的公共方法
    fun getRandomNumber(): Int {
        return generator.nextInt()
    }
}

Client端要做的事

  1. 建立ServiceConnection類型實例並覆載onServiceConnected()方法和onServiceDisconnected()方法。
  2. 當執行到onServiceConnected callback時可通過IBinder instance得到Service的instance這樣可實現client與Service的連接。
  3. onServiceDisconnected callback被執行時表示client與Service已經斷開在此可以寫一些斷開後需要做的處理。
class AActivity : AppCompatActivity() {

    private var service: MyService? = null
    private var isBind = false

    private var conn = object : ServiceConnection{
        override fun onServiceConnected(p0: ComponentName?, p1: IBinder?) {
            isBind = true
            val myBinder = p1 as MyService.MyBinder
            service = myBinder.service
            Log.i("xiao", "ActivityA - onServiceConnected")
            val num = service!!.getRandomNumber()
            Log.i("xiao", "ActivityA - getRandomNumber = $num");
        }

        override fun onServiceDisconnected(p0: ComponentName?) {
            isBind = false
            Log.i("xiao", "ActivityA - onServiceDisconnected")
        }
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_a)

        Log.i("xiao", "ActivityA - onCreate - Thread = " + Thread.currentThread().name)

        btn_bind_service_a.setOnClickListener {
            val intent = Intent(this,MyService::class.java)
            intent.putExtra("from","ActivityA")
            Log.i("xiao", "----------------------------------------")
            Log.i("xiao", "ActivityA 执行 bindService");
            bindService(intent, conn, Context.BIND_AUTO_CREATE)
        }
        btn_unbind_service_a.setOnClickListener {
            if(isBind){
                Log.i("xiao", "----------------------------------------")
                Log.i("xiao", "ActivityA 执行 unbindService");
                unbindService(conn)
            }
        }
        btn_a_start_b.setOnClickListener {
            val intent = Intent(this,BActivity::class.java)
            Log.i("xiao", "----------------------------------------")
            Log.i("xiao", "ActivityA 启动 ActivityB");
            startActivity(intent)
        }
        btn_finish_a.setOnClickListener {
            Log.i("xiao", "----------------------------------------")
            Log.i("xiao", "ActivityA 执行 finish");
            finish()
        }
    }

    override fun onDestroy() {
        super.onDestroy()
        Log.i("xiao", "ActivityA - onDestroy")
    }

}

參考

  1. 如何使用Service(kotlin)
  2. Android kotlin service使用简析 - 簡書
  3. 《Android》『Service』- 背景執行服務的基本用法 - 賽肥膩膩の娛樂生活誌