Service 的开发精要

Service的2个分类以及各自的生命周期
service.png

通过startService() 启动的服务叫做本地服务, 即Local Service, 客户端只是能启动service, service一般新建线程干自己的事, 无法调用service的方法进行交互.
通过bindService() 启动的服务叫做远程服务,即Remote Service, 也叫做基于aidl的服务, 支持客户端和服务端可以相互调用对方的方法.

关键方法暗示的信息

小技巧: 看继承自Service的类时,如果onBind()返回null, 那就知道这个Service就是本地服务, 客户端只能通过startService()启动, 不能通过bindService() 启动.

eg.
PrecacheService.java
public class PrecacheService extends Service {
    /** PrecacheService does not support binding. */
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
}
本地服务的启动和停止, 以及int onStartComand()的返回值.

启动:

Intent intent = new Intent(context, DownloadService.class);
context.startService(intent);

停止:

Intent intent = new Intent(context, DownloadService.class);
context.stopService(intent);

或 Service自己调用stopSelf() API.

    private void shutdownPrecaching() {
        mIsPrecaching = false;
        releasePrecachingWakeLock();
        stopSelf();
    }

int onStartComand()的返回值含义:

START_STICKY:
包含Service的进程被异常kill掉,系统会自动重启该服务, 但是传递给onStartComand()的intent为null.

START_NOT_STICKY:
包含Service的进程被异常kill掉,系统不会自动重启该服务.

START_REDELIVER_INTENT:
包含Service的进程被异常kill掉,系统会自动重启该服务, 并且把之前保留的intent传递给onStartComand().

另外:

START_STICKY和 START_NOT_STICKY:当进程被杀死后onDestroy()是不会被执行的!
START_REDELIVER_INTENT :当进程被杀死后onDestroy()会被执行!
远程服务的基本使用方法

客户端和服务端共享完全相同内容的aidl文件,里面定义服务端暴露给客户端可以调用的API, 各自通过IDE生成其对应的java文件.
服务端代码:

  1. 在Service中, 声明一个
private Binder mBinder = new IBookManager.Stub() {
//实现对aidl中的方法
//IBookManager是通过aidl生成的class, Stub是生成class中的内部类
}
  1. 在onBind()方法中把这个mBinder返回给客户端使用.
@Override
public IBinder onBind(Intent intent) {
    return mBinder;
}

客户端代码:

  1. 创建一个实现了ServiceConnection接口的对象, 实现里面的2个方法,
    onServiceConnected()和onServiceDisconnected().
    在onServiceConnected()中把服务端返回的mBinder对象, 转换为aidl生成类的对象, 之后就可以通过这个对象访问服务端方法了.
public void onServiceConnected(ComponentName className, IBinder service) {
    IBookManager bookManager = IBookManager.Stub.asInterface(service);
    mRemoteBookManager = bookManager;
}
其中, IBookManager就是aidl文件生成的类.
  1. 使用ServiceConnection的对象和Intent对象, 连接服务端Service.
Intent intent = new Intent(this, BookManagerService.class);
//绑定Service, 成功后ServiceConnection mConnection的onServiceConnected()方法被回调.
bindService(intent, mConnection, Context.BIND_AUTO_CREATE); 
  1. 解除绑定调用unbindService()
context.unbindService(mConnection);
不要在onServiceConnected中直接调用远程方法

因为onServiceConnected()运行在UI线程, 调用的远程方法可能是个耗时操作,无法预知,
因此不能直接调用, 避免ANR的发生.

如何实现服务端调用客户端的方法

使用RemoteCallbackList类
在aidl中提供注册API, 客户端把自己的一个对象传递给服务端,
服务端通过RemoteCallbackList类型的一个对象保存所有的客户端对象,
通过遍历RemoteCallbackList对象就可以调用客户端的方法.

private RemoteCallbackList<IOnNewBookArrivedListener> mListenerList

这样就实现了双向调用.

服务端意外停止后, 客户端应该如何处理

创建一个IBinder.DeathRecipient接口的对象, 并实现里面的binderDied()方法.
在客户端里去调用IBinder的linkToDeath()注册,
这样服务端意外中止后, binderDied()方法会被回调.

private IBinder.DeathRecipient mDeathRecipient = new IBinder.DeathRecipient() {
@Override
public void binderDied() {
    Log.d(TAG, "binder died. tname:" + Thread.currentThread().getName());
    if (mRemoteBookManager == null)
        return;
    mRemoteBookManager.asBinder().unlinkToDeath(mDeathRecipient, 0);
    mRemoteBookManager = null;
    // TODO:这里重新绑定远程Service
}
};

mRemoteBookManager.asBinder().linkToDeath(mDeathRecipient, 0);

同时, 客户端的onServiceDisconnected()在服务端意外中止时也会被回调,
只是区别在于onServiceDisconnected()运行在UI线程, binderDied()运行在Binder线程池中的线程.
服务端意外中止时, 客户端的处理方式一般是重新去bindService().

权限认证

保证服务端的service只准许被公司内部应用连接.

  1. 可以在Service的onBind()中调用checkCallingOrSelfPermission()检查客户端的AndroidManifest.xml是否声明了特定权限
@Override
public IBinder onBind(Intent intent) {
    int check = checkCallingOrSelfPermission("com.ryg.chapter_2.permission.ACCESS_BOOK_SERVICE");
    Log.d(TAG, "onbind check=" + check);
    if (check == PackageManager.PERMISSION_DENIED) {
        return null;
    }
    return mBinder;
}
<uses-permission android:name="com.ryg.chapter_2.permission.ACCESS_BOOK_SERVICE"/>

检查没有权限的话, 返回null.

2.也可以在服务端的onTransact()中检查权限和客户端的uid是否是公司内部应用
查看通过aidl生成的java类, 可以看到所有客户端调用服务端的方法都是首先进入到onTransact(),
在onTransact()中再通过switch case调用到真正的aidl中的方法,所以可以通过复写onTransact()进行权限认证.

//双重安全性检查.
Override
public boolean onTransact(int code, Parcel data, Parcel reply, int flags)
        throws RemoteException {
    int check = checkCallingOrSelfPermission("com.ryg.chapter_2.permission.ACCESS_BOOK_SERVICE");
    Log.d(TAG, "check=" + check);
    if (check == PackageManager.PERMISSION_DENIED) {
        return false;
    }

    //通过getCallingUid()得到客户端的uid, 再通过PackageManager根据uid查到package name进行检查.
    String packageName = null;
    String[] packages = getPackageManager().getPackagesForUid(
            getCallingUid());
    if (packages != null && packages.length > 0) {
        packageName = packages[0];
    }
    Log.d(TAG, "onTransact: " + packageName);
    if (!packageName.startsWith("com.ryg")) {
        return false;
    }

    //验证都通过后,才在super.onTransact()中调用真正的aidl方法.
    return super.onTransact(code, data, reply, flags);
}
项目中的应用场景

本地服务: DownloadService.java 实现下载功能
远程服务: IUIAdapter.aidl 实现调用手机助手的方法

IntentService

作为本地服务来说, Service的几个生命周期方法都是运行在UI线程,
所以通常的做法是再开一个新线程执行实际的工作. 为简化这个操作,
framework提供了IntentService, 关键方法是onHandleIntent(Intent intent),
在一个worker thread中处理传进来的intent.
使用IntentService的目的是为了简化直接继承Service作为本地服务时的操作.

chromium的MinidumpUploadService extends IntentService,
用于上传造成crash时的dump文件.

代码实践

完整的示例代码可以看之前的总结:
http://www.jianshu.com/p/7e97076d8613 (IPC机制)
https://github.com/singwhatiwanna/android-art-res/blob/master/Chapter_2/src/com/ryg/chapter_2/aidl/BookManagerService.java
自己写了一个app, 用于监测指定进程的cpu占用情况, 并实现了双向调用操作.
完整app代码在:
https://github.com/AandK/PerformanceMonitor

生命周期的调用参考图:
http://www.th7.cn/Program/Android/201411/307510.shtml

------------DONE---------

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

推荐阅读更多精彩内容