Android四大组件之Service

什么是Service

  1. Service是一个应用组件, 它用来在后台完成一个时间跨度比较大的工作且没有关联任何界面
  2. 一个Service可以完成下面这些工作:
    • 访问网络
    • 播放音乐
    • 文件IO操作
    • 大数据量的数据库操作……
  3. 服务的特点:
    • Service在后台运行,不用与用户进行交互
    • 即使应用退出, 服务也不会停止
    • 在默认情况下,Service运行在应用程序进程的主线程(UI线程)中,如果需要在Service中处理一些网络连接等耗时的操作,那么应该将这些任务放在分线程中处理,避免阻塞用户界面

Service的分类

  • Local Service(本地服务):Service对象与Serive的启动者在同个进程中运行, 两者的通信是进程内通信,通俗点就是启动同一应用的Service就是本地服务。

  • Remote Service(远程服务):Service对象与Service的启动者不在同一个进程中运行, 这时存在一个进程间通信的问题, Android专门为此设计了AIDL来实现进程间通信。通俗点就是启动其他应用的Service就是远程服务。

Service和Thread区别

  1. Service:

    • 用来在后台完成一个时间跨度比较大的工作的应用组件
    • Service的生命周期方法运行在主线程, 如果Service想做持续时间比较长的工作, 需要启动一个分线程(Thread)
    • 应用退出: Service不会停止
    • 应用再次进入: 可以与正在运行的Service进行通信
  2. Thread:

    • 用来开启一个分线程的类, 做一个长时间的工作
    • Thread对象的run()在分线程执行
    • 应用退出: Thread不会停止,
    • 应用再次进入: 不能再控制前面启动的Thread对象

定义Service

  1. 定义一个类继承于Service类

    public class MyService extends Service {
        
        @Nullable
        @Override
        public IBinder onBind(Intent intent) {
            return null;
        }
        
        ......
    }
    
  2. 在AndroidManifest.xml中配置Service

    <service android:name=".MyService">
        <intent-filter>
            <action android:name="guo.ping.activitydemo.MyService"/>
        </intent-filter>
    </service>
    

启动、停止Service

  • 一般启动与停止

    context.startService(Intent intent)
    context.stopService(Intent intent)
    
  • 绑定启动与解绑

    context.bindService(Intent intent, ServiceConnection connection)
    context.unbindService(ServiceConnection connection)
    
  • 一般启动停止Service

    public void startMyService(View view){
        Intent intent = new Intent(this, MyService.class);
        startService(intent);
    }
    
    public void stopMyService(View view){
        Intent intent = new Intent(this, MyService.class);
        stopService(intent);
    }
    
  • 绑定启动Service与解绑:

    public void bindMyService(View view) {
        Intent intent = new Intent(this, MyService.class);
    
        if (mServiceConnection == null) {
            mServiceConnection = new ServiceConnection() {
                @Override
                public void onServiceConnected(ComponentName name, IBinder service) {
                    Log.i(TAG, "onServiceConnected()" + service.hashCode());
                }
    
                @Override
                public void onServiceDisconnected(ComponentName name) {
                    Log.i(TAG, "onServiceDisconnected()");
                }
            };
    
            bindService(intent, mServiceConnection, Context.BIND_AUTO_CREATE);
        } else {
            Log.i(TAG, "已经绑定...");
        }
    }
    
    public void unbindMyService(View view) {
        if (mServiceConnection != null) {
            Intent intent = new Intent(this, MyService.class);
            unbindService(mServiceConnection);
    
            mServiceConnection = null;
        } else {
            Log.i(TAG, "已经解绑...");
        }
    }
    

    MyService类重写onBind()和onUnbind()方法:

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        Log.i(TAG, "MyService onBind()");
        Binder binder = new Binder();
        Log.i(TAG, "binder.hashCode() = " + binder.hashCode());
        return binder;
    }
    
    @Override
    public boolean onUnbind(Intent intent) {
        Log.i(TAG, "MyService onUnbind()");
        return super.onUnbind(intent);
    }
    

    测试结果如下,可以看到在MyService的onBind()方法中返回的Binder对象就是在onServiceConnected()回调方法中的IBinder类型的形参service。

    07-29 10:52:43.002 2794-2794/guo.ping.activitydemo I/MyService: MyService()
    07-29 10:52:43.002 2794-2794/guo.ping.activitydemo I/MyService: MyService onCreate()
    07-29 10:52:43.002 2794-2794/guo.ping.activitydemo I/MyService: MyService onBind()
    07-29 10:52:43.002 2794-2794/guo.ping.activitydemo I/MyService: 1394383824
    07-29 10:52:43.002 2794-2794/guo.ping.activitydemo I/MainActivity: onServiceConnected()1394383824
    07-29 10:52:46.854 2794-2794/guo.ping.activitydemo I/MyService: MyService onUnbind()
    07-29 10:52:46.854 2794-2794/guo.ping.activitydemo I/MyService: MyService onDestroy()
    

    这种方式下,Activity和Service是通过ServiceConnection相关联的,如果Activity销毁则这个链接是要断开的,会引起ServiceConnectionLeaked异常:

    07-29 11:00:39.046 2794-2794/guo.ping.activitydemo E/ActivityThread: Activity guo.ping.activitydemo.MainActivity has leaked ServiceConnection guo.ping.activitydemo.MainActivity$1@5320601c that was originally bound here
    
    android.app.ServiceConnectionLeaked: Activity guo.ping.activitydemo.MainActivity has leaked ServiceConnection guo.ping.activitydemo.MainActivity$1@5320601c that was originally bound here
    

    需要我们在Activity的onDestory方法中进行解绑操作:

    @Override
    protected void onDestroy() {
        super.onDestroy();
        Log.i(TAG, "onDestroy()");
        if(mServiceConnection != null){
            Intent intent = new Intent(this, MyService.class);
            unbindService(mServiceConnection);
    
            mServiceConnection = null;
        }
    }
    

Service的生命周期

Service的生命周期
  • 当用一般方式启动Service时,如果Service已经启动,则不会调用onCreated(),直接调用onStartCommand()方法;
  • 当有多个Activity等组件和一个Service进行绑定,必须所有的客户端全部调用unbindService()方法进行解绑Service端才能调用onUnbind()方法,这一点还是合乎情理的。

AIDL

  • 每个应用程序都运行在自己的独立进程中,并且可以启动另一个应用进程的服务,而且经常需要在不同的进程间传递数据对象。
  • 在Android平台,一个进程不能直接访问另一个进程的内存空间,所以要想对话,需要将对象分解成操作系统可以理解的基本单元,并且有序的通过进程边界。
  • 如果在一个进程中(例如Activity)要调用另一个进程中(例如Service)对象的操作,就可以使用AIDL生成可序列化的参数。
  • AIDL (Android Interface Definition Language):用于生成可以在Android设备上两个进程之间进行进程间通信(interprocess communication, IPC)的代码。

使用AIDL的思路理解

整个的流程其实大致是,在本进程(应用)中,实现某一需求需要调用另一个进程(应用)的某个服务(服务中包含某个具体的方法),所以我们需要先将请求参数发给另一个进程的远程服务,然后远程服务将参数接收调用方法处理得出结果,并将结果返回给原来的进程。而AIDL就是充当这个传输数据的角色,将参数送至、结果传回。


AIDL整体流程

AIDL文件的书写规则

  1. 接口名和aidl文件名相同.
  2. 接口和方法前不用加访问权限修饰符public,private,protected等,也不能用final,static.
  3. Aidl默认支持的类型包话java基本类型(int,long,boolean等)和(String,List,Map, CharSequence),使用这些类型时不需要import声明.对于List和Map中的元素类型必须是Aidl支持的类型.如果使用自定义类型作为参数或返回值,自定义类型必须实现Parcelable接口.
  4. 自定义类型和AIDL生成的其它接口类型在aidl描述文件中,应该显式import,即便在该类和定义的包在同一个包中.
  5. 在aidl文件中所有非Java基本类型参数必须加上in、out、inout标记,以指明参数是输入参数、输出参数还是输入输出参数.
  6. Java原始类型默认的标记为int,不能为其它标记.

使用AIDL的Demo

推荐博客:

需求:

  • 传递俩个int类型数字a、b,调用远程方法计算俩者之和并返回
  • 通过id来调用远程服务查找对应的学生记录并返回

实现:

  1. 定义Student的JavaBean文件,需要实现Parcelable接口。在打包与解包时,顺序要一致,否则会出现错误;

    package guo.ping.testservice;
    
    import android.os.Parcel;
    import android.os.Parcelable;
    
    public class Student implements Parcelable {
    
        private int id;
        private String name;
        private double weight;
        
        public Student() {
        }
        
        public Student(int id, String name, double weight) {
            this.id = id;
            this.name = name;
            this.weight = weight;
        }
    
        public int getId() {
            return id;
        }
    
        public void setId(int id) {
            this.id = id;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public double getWeight() {
            return weight;
        }
    
        public void setWeight(double weight) {
            this.weight = weight;
        }
    
        @Override
        public String toString() {
            return "Student{" +
                    "id=" + id +
                    ", name='" + name + '\'' +
                    ", weight=" + weight +
                    '}';
        }
    
        // 添加一个静态成员,名为CREATOR,该对象实现了Parcelable.Creator接口
        public static final Parcelable.Creator<Student> CREATOR = new Creator<Student>() {
            @Override
            public Student createFromParcel(Parcel source) {
                int id = source.readInt();
                String name = source.readString();
                double weight = source.readDouble();
                return new Student(id, name, weight);
            }
    
            @Override
            public Student[] newArray(int size) {
                return new Student[size];
            }
        };
    
        @Override
        public int describeContents() {
            return 0;
        }
    
        //将当前对象的属性数据写到Parcel包对象中(也就是打包)
        @Override
        public void writeToParcel(Parcel dest, int flags) {
            dest.writeInt(id);
            dest.writeString(name);
            dest.writeDouble(weight);
        }
    }
    
  2. 定义Student.aidl文件。文件名只能是Student,其他会出错;

    package guo.ping.testservice;
    
    parcelable Student;
    
  3. 定义IAdd.aidl文件,这个文件里面包含着我们需要远程服务包含的抽象方法;

    package guo.ping.testservice;
    import guo.ping.testservice.Student;
    
    interface IAdd {
        int add(int a, int b);
        Student queryStudentById(int id);
    }
    
  4. 将上述三个文件拷贝至另一个工程,总之客户端与服务端都需要拥有,包名要一致,否则也是出错;

    工程目录结构
  5. AndroidStudio中选择Build——>Make Project,生成AIDL的代码;

  6. 在服务端编写Service,并实现AIDL文件中的抽象方法;

    public class MyRemoteService extends Service {
    
        private IBinder mIBinder = new IAdd.Stub() {
            @Override
            public int add(int a, int b) throws RemoteException {
                return a + b;
            }
    
            @Override
            public Student queryStudentById(int id) throws RemoteException {
                return new Student(id, "假装查了数据库叫Tom", 140.5);
            }
        };
    
    
        @Nullable
        @Override
        public IBinder onBind(Intent intent) {
            return mIBinder;
        }
    
        @Override
        public boolean onUnbind(Intent intent) {
            return super.onUnbind(intent);
        }
    }
    

    这里定义了成员变量mIBinder,new IAdd.Stub()是Android依据我们编写的AIDL文件生成的Java代码里面写好的。然后就是注册Service,这里需要intent-filter定义好便于隐式意图启动。

    <service android:name=".MyRemoteService">
        <intent-filter>
            <action android:name="guo.ping.testservice.MyRemoteService"/>
        </intent-filter>
    </service>
    
  7. 客户端编写启动远程Service的方法;

    public void callTheRemoteServiceMethod(View view) throws RemoteException {
        int add = mIAdd.add(2, 5);
        Log.i(TAG, add + "");
    
        Student student = mIAdd.queryStudentById(10);
        Log.i(TAG, student.toString());
    }
    
    
    public void startRemoteService(View view) {
        Intent intent = new Intent("guo.ping.testservice.MyRemoteService");
    
        if (mServiceConnection == null) {
            mServiceConnection = new ServiceConnection() {
                @Override
                public void onServiceConnected(ComponentName name, IBinder service) {
                    mIAdd = IAdd.Stub.asInterface(service);
                    Log.i(TAG, "onServiceConnected()");
                }
    
                @Override
                public void onServiceDisconnected(ComponentName name) {
                    Log.i(TAG, "onServiceDisconnected()");
                }
            };
    
            bindService(intent, mServiceConnection, Context.BIND_AUTO_CREATE);
        } else {
            Log.i(TAG, "已经绑定...");
        }
    }
    
    public void stopRemoteService(View view) {
    
        if (mServiceConnection != null) {
            unbindService(mServiceConnection);
            mIAdd = null;
            mServiceConnection = null;
        } else {
            Log.i(TAG, "已经解绑...");
        }
    }
    
  8. 测试结果如下;

    测试结果

补充

在关闭远程服务的时候,发现重写的onServiceDisconnected()方法Log不打印,想着onServiceConnected()要打印必须在Service的onBind()方法中返回IBinder对象,所以便在Service中实现onUnbind()方法,发现并没有乱用。
百度了一下,原来当service所在进程crash或者被kill的时候,onServiceDisconnected才会被呼叫。具体参考:我的onServiceDisconnected为什么没有被呼叫

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

推荐阅读更多精彩内容

  • Service的生命周期 service的生命周期,从它被创建开始,到它被销毁为止,可以有两条不同的路径: A s...
    _执_念__阅读 1,549评论 0 19
  • 1、什么是service 一个可以在后台执行长时间操作而不需要用户界面的应用组件。 需要注意的是: - 服务并不是...
    hahauha阅读 581评论 0 1
  • 服务基本上分为两种形式 启动 当应用组件(如 Activity)通过调用 startService() 启动服务时...
    pifoo阅读 1,265评论 0 8
  • Service的相关知识虽然简单,但是也比较琐碎,其衍生知识也比较多。本篇从Service的生命周期、运行和使用方...
    卑鄙的鹿尤菌阅读 803评论 0 1
  • 文字有时也仅仅是为了纪念。 前天是我三十八周岁生日,请老母亲和姐姐一家吃饭,选了新开业的蒸汽海鲜,据说是当下流行的...
    Marktoo阅读 383评论 0 0