Service 的两种启动方式和「Service 与 Activity 数据交互」

1. Service 的两种启动方式

Activity 中可以有两种方式启动 Service,不同方式启动时 Service 的生命周期也不一样,现在在 Activity 中定义四个 Button,分别是 startServicestopServicebindServiceunbindService,Service 中各生命周期中分别打印 Log 日志,通过日志查看生命周期执行情况:

// MainActivity.kt
class MainActivity : AppCompatActivity(){
    var mService: MyService? = null
    var isBind = false
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        val intent = Intent(this, MyService::class.java)
        val conn = object : ServiceConnection {
            override fun onServiceDisconnected(name: ComponentName?) {
                Log.e("abc", "-- Activity 中 onServiceDisconnected --")
            }

            override fun onServiceConnected(name: ComponentName?, service: IBinder?) {
                mService = (service as MyService.MyBinder).getService()
                Log.e("abc", "-- Activity 中 onServiceConnected --")
            }
        }

        startService.setOnClickListener {
            startService(intent)
        }
        stopService.setOnClickListener {
            stopService(intent)
        }
        bindService.setOnClickListener {
            isBind = bindService(intent, conn, Context.BIND_AUTO_CREATE)
        }
        unBindService.setOnClickListener {
            if (isBind) {
                isBind = false
                unbindService(conn)
            }
        }
    }
}
// MyService.kt
class MyService: Service() {
    override fun onBind(intent: Intent?): IBinder? {
        Log.e("abc", "-- onBind --")
        return MyBinder()
    }

    override fun onCreate() {
        super.onCreate()
        Log.e("abc", "-- onCreate --")
    }

    override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
        Log.e("abc", "-- onStartCommand --")
        return super.onStartCommand(intent, flags, startId)
    }

    override fun onDestroy() {
        super.onDestroy()
        Log.e("abc", "-- onDestroy --")
    }

    override fun onUnbind(intent: Intent?): Boolean {
        Log.e("abc", "-- onUnbind --")
        return super.onUnbind(intent)
    }

    inner class MyBinder : Binder() {
        fun getService(): MyService {
            return this@MyService
        }
    }
}

其实以前 Service 还有个生命周期叫 onStart(),后来被弃用了,它的功能由onStartCommand()代替。

1.1 startService

在 MainActivity 中可以这样启动和停止 Service:

// 启动
val intent = Intent(this, MyService::class.java)
startService(intent)
// 停止
stopService(intent)
// Log
-- onCreate --
-- onStartCommand --
-- onDestroy --

1.2 bindService

bindService 方式稍微复杂些:

// 绑定
val conn = object : ServiceConnection {
    override fun onServiceDisconnected(name: ComponentName?) {}
    override fun onServiceConnected(name: ComponentName?, service: IBinder?) { }
}
bindService(intent, conn, Context.BIND_AUTO_CREATE)
// 停止
unbindService(conn)
// Log
-- onCreate --
-- onBind --
-- onUnbind --
-- onDestroy --

绑定时会执行 onBind 生命周期,解绑时先调用 onUnbind再调用onDestroy

1.3 两种方式混合

如果先用 startService 方式启动了 Service,再用 bindService 绑定一次(两者顺序也可以颠倒),那么此时单纯的 stopService 或者 unbindService 是无法终止 Service 的,需要二者联合使用才行。具体细化:

1.3.1 先 stopService 后unbindService

// Log
-- onCreate --
-- onStartCommand --
-- onBind --

// stopService 无反应

// unbindService:
-- onUnbind --
-- onDestroy --

调用 stopService 没有反应,调用 unbindService 时方可销毁 Service。

1.3.2 先 unbindService 后 stopService

-- onCreate --
-- onStartCommand --
-- onBind --

// unbindService
-- onUnbind --

// stopService
-- onDestroy --

调用 unbindService 只能解绑(onUnbind)不能销毁,调用 stopService 时才可以销毁 Service

1.4 注意事项

  1. 多次调用 startService 时,Service 中的 onStartCommand方法会执行多次;但多次使用 bindService 时,onBind 只执行一次。
  2. bindService 方式打开 Service 时,Service 的生命周期是和打开它的 Activity 绑定的,而 startService 方式打开的 Service 在 Activity 被销毁后(onDestroy),还可以继续存活(可以同时打印 Activity 和 Service 的生命周期查看,这里不举例子了)。

2. Service 与 Activity 数据交互

其实从前面的代码中也可以看出,在 MainActivity 中,可以获取到 Service 的引用(具体来说是onServiceConnected中),Service 调用 Activity 中的方法主要讲讲。

2.1 Activity 调用 Service 方法

回看 Service 的onBind生命周期可以发现,该方法返回的是一个 IBinder 类型,我们在具体实现是返回的是它子类的子类(IBinder 的 子类是 Binder,Binder 的子类是 MyBinder),这是一种多态(不懂多态的自己查一下蛤,这篇文章不是介绍多态的,展开说太长了):

// MyBinder 是 Service 我自己定义的里面的内部类
inner class MyBinder : Binder() {
        fun getService(): MyService {
            return this@MyService
        }
    }
// Binder 源码
public class Binder implements IBinder {
...代码省略
}
// onBind 生命周期
override fun onBind(intent: Intent?): IBinder? {
        return MyBinder()
    }
// bindService 方式打开 Service 时用到的 conn
val conn = object : ServiceConnection {
            override fun onServiceDisconnected(name: ComponentName?) {
            }
            override fun onServiceConnected(name: ComponentName?, service: IBinder?) {
                mService = (service as MyService.MyBinder).getService()
            }
        }

现在,可以用 mService 对象调用 Service 中的任何方法,比如 printTxt

 fun printTxt(txt : String) {
        Log.e("abc", "-- this is txt -- $txt")
    }

其实我觉得还可以直接 new 一个 MyService 对象 mService2,然后 myService2.printTxt("装逼"),好像也没啥区别。

2.2 Service 主动向 Activity 传输数据

Service 没有 Activity 的引用,所以可以通过接口回调或者广播的方式向 Activity 传递数据。

2.2.1 接口回调

定义接口:

interface CallBack {
    fun call(index: Int)
}

Service 中初始化:

private var callBack:CallBack ?= null
fun setListener(callBack: CallBack){
    this.callBack = callBack
}

Activity 实现 CallBack(或者用 匿名内部类):

class MainActivity : AppCompatActivity(), CallBack {
    override fun call(index: Int) {
        Log.e("abc", "This is index value : $index")
    }
}
mService.setListener(this@MainActivity)

// 或者
mService!!.setListener(object :CallBack{
    override fun call(index: Int) {
        Log.e("abc", "This is index value : $index")
    }
})

Service 中有了 callBack 对象,就可以主动向 Activity 传输数据了。

2.2.2 广播

用 Android 自带的 BroadcastReceiver 或者 EventBus 这种第三方框架区别不大,先在 Activity 中注册接受者,然后在 Service 中发射广播数据,不具体举例了。需要说明的是,如果是一个 Service 向多个 Activity 传递数据,广播比回调要好一些。

源码地址

点击查看

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容