Android应用开发基础之广播与服务

广播

  • 广播的概念
    • 现实:电台通过发送广播发布消息,买个收音机,就能收听
    • Android:系统在产生某个事件时发送广播,应用程序使用广播接收者接收这个广播,就知道系统产生了什么事件。 Android系统在运行的过程中,会产生很多事件,比如开机、电量改变、收发短信、拨打电话、屏幕解锁

IP拨号器

原理:接收拨打电话的广播,修改广播内携带的电话号码 * 定义广播接收者接收打电话广播

public class CallReceiver extends BroadcastReceiver {

    //当广播接收者接收到广播时,此方法会调用
    @Override
    public void onReceive(Context context, Intent intent) {
        //拿到用户拨打的号码
        String number = getResultData();
        //修改广播内的号码
        setResultData("17951" + number);
    }
}
  • 在清单文件中定义该广播接收者接收的广播类型

    <receiver android:name="com.ibky.ipdialer.CallReceiver">
        <intent-filter >
            <action android:name="android.intent.action.NEW_OUTGOING_CALL"/>
        </intent-filter>
    </receiver>
    
  • 接收打电话广播需要权限

    <uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS"/>
    
  • 即使广播接收者的进程没有启动,当系统发送的广播可以被该接收者接收时,系统会自动启动该接收者所在的进程

短信拦截器

系统收到短信时会产生一条广播,广播中包含了短信的号码和内容

  • 定义广播接收者接收短信广播

    public void onReceive(Context context, Intent intent) {
    //拿到广播里携带的短信内容
    Bundle bundle = intent.getExtras();
    Object[] objects = (Object[]) bundle.get("pdus");
    for(Object ob : objects ){
        //通过object对象创建一个短信对象
        SmsMessage sms = SmsMessage.createFromPdu((byte[])ob);
        System.out.println(sms.getMessageBody());
        System.out.println(sms.getOriginatingAddress());
    }
    

    }

  • 系统创建广播时,把短信存放到一个数组,然后把数据以pdus为key存入bundle,再把bundle存入intent
  • 清单文件中配置广播接收者接收的广播类型,注意要设置优先级属性,要保证优先级高于短信应用,才可以实现拦截

    <receiver android:name="com.ibky.smslistener.SmsReceiver"> <intent-filter android:priority="1000"> <action android:name="android.provider.Telephony.SMS_RECEIVED"/> </intent-filter> </receiver> 
  • 添加权限

    <uses-permission android:name="android.permission.RECEIVE_SMS"/>
    
  • 4.0以后广播接收者安装以后必须手动启动一次,否则不生效

  • 4.0以后广播接收者如果被手动关闭,就不会再启动了

监听SD卡状态

  • 清单文件中定义广播接收者接收的类型,监听SD卡常见的三种状态,所以广播接收者需要接收三种广播

     <receiver android:name="com.ibky.sdcradlistener.SDCardReceiver">
        <intent-filter >
            <action android:name="android.intent.action.MEDIA_MOUNTED"/>
            <action android:name="android.intent.action.MEDIA_UNMOUNTED"/>
            <action android:name="android.intent.action.MEDIA_REMOVED"/>
            <data android:scheme="file"/>
        </intent-filter>
    </receiver>
    
  • 广播接收者的定义

    public class SDCardReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            // 区分接收到的是哪个广播
            String action = intent.getAction();
    
            if(action.equals("android.intent.action.MEDIA_MOUNTED")){
                System.out.println("sd卡就绪");
            }
            else if(action.equals("android.intent.action.MEDIA_UNMOUNTED")){
                System.out.println("sd卡被移除");
            }
            else if(action.equals("android.intent.action.MEDIA_REMOVED")){
                System.out.println("sd卡被拔出");
            }
        }
    }
    

勒索软件

  • 接收开机广播,在广播接收者中启动勒索的Activity
  • 清单文件中配置接收开机广播

    <receiver android:name="com.ibky.lesuo.BootReceiver">
        <intent-filter >
            <action android:name="android.intent.action.BOOT_COMPLETED"/>
        </intent-filter>
    </receiver>
    
  • 权限

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
    
  • 定义广播接收者

    @Override
    public void onReceive(Context context, Intent intent) {
        //开机的时候就启动勒索软件
        Intent it = new Intent(context, MainActivity.class);        
        context.startActivity(it);
    }
    
  • 以上代码还不能启动MainActivity,因为广播接收者的启动,并不会创建任务栈,那么没有任务栈,就无法启动activity
  • 手动设置创建新任务栈的flag

    it.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    

监听应用的安装、卸载、更新

原理:应用在安装卸载更新时,系统会发送广播,广播里会携带应用的包名 * 清单文件定义广播接收者接收的类型,因为要监听应用的三个动作,所以需要接收三种广播

    <receiver android:name="com.ibky.app.AppReceiver">
        <intent-filter >
            <action android:name="android.intent.action.PACKAGE_ADDED"/>
            <action android:name="android.intent.action.PACKAGE_REPLACED"/>
            <action android:name="android.intent.action.PACKAGE_REMOVED"/>
            <data android:scheme="package"/>
        </intent-filter>
    </receiver>
  • 广播接收者的定义

    public void onReceive(Context context, Intent intent) {
        //区分接收到的是哪种广播
        String action = intent.getAction();
        //获取广播中包含的应用包名
        Uri uri = intent.getData();
        if(action.equals("android.intent.action.PACKAGE_ADDED")){
            System.out.println(uri + "被安装了");
        }
        else if(action.equals("android.intent.action.PACKAGE_REPLACED")){
            System.out.println(uri + "被更新了");
        }
        else if(action.equals("android.intent.action.PACKAGE_REMOVED")){
            System.out.println(uri + "被卸载了");
        }
    }
    

广播的两种类型

  • 无序广播:所有跟广播的intent匹配的广播接收者都可以收到该广播,并且是没有先后顺序(同时收到)
  • 有序广播:所有跟广播的intent匹配的广播接收者都可以收到该广播,但是会按照广播接收者的优先级来决定接收的先后顺序
    • 优先级的定义:-1000~1000
    • 最终接收者:所有广播接收者都接收到广播之后,它才接收,并且一定会接收
    • abortBroadCast:阻止其他接收者接收这条广播,类似拦截,只有有序广播可以被拦截

Service

  • 就是默默运行在后台的组件,可以理解为是没有前台的activity,适合用来运行不需要前台界面的代码
  • 服务可以被手动关闭,不会重启,但是如果被自动关闭,内存充足就会重启
  • startService启动服务的生命周期
    • onCreate-onStartCommand-onDestroy
  • 重复的调用startService会导致onStartCommand被重复调用

进程优先级

  1. 前台进程:拥有前台activity(onResume方法被调用)
  2. 可见进程:拥有可见activity(onPause方法被调用)
  3. 服务进程:不到万不得已不会被回收,而且即便被回收,内存充足时也会被重启
  4. 后台进程:拥有后台activity(activity的onStop方法被调用了),很容易被回收
  5. 空进程:没有运行任何activity,很容易被回收

电话窃ting器

  • 电话状态:空闲、响铃、接听
  • 获取电话管理器,设置侦听

    TelephonyManager tm = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);
    tm.listen(new MyPhoneStateListener(), PhoneStateListener.LISTEN_CALL_STATE);
    
  • 侦听对象的实现

    class MyPhoneStateListener extends PhoneStateListener{
    
        //当电话状态改变时,此方法调用
        @Override
        public void onCallStateChanged(int state, String incomingNumber) {
            // TODO Auto-generated method stub
            super.onCallStateChanged(state, incomingNumber);
            switch (state) {
            case TelephonyManager.CALL_STATE_IDLE://空闲
                if(recorder != null){
                    recorder.stop();
                    recorder.release();
                }
                break;
            case TelephonyManager.CALL_STATE_OFFHOOK://摘机
                if(recorder != null){
                    recorder.start();
                }
                break;
            case TelephonyManager.CALL_STATE_RINGING://响铃
                recorder = new MediaRecorder();
                //设置声音来源
                recorder.setAudioSource(MediaRecorder.AudioSource.MIC);
                //设置音频文件格式
                recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
                recorder.setOutputFile("sdcard/haha.3gp");
                //设置音频文件编码
                recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
                try {
                    recorder.prepare();
                } catch (IllegalStateException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                } catch (IOException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
                break;
            }
        }
    }
    

随堂笔记

Android四大组件

  • Activity
  • BroadcastReceiver
  • service
  • ContentProvider

广播

  • 现实
    • 电台通过播放广播,发布消息
    • 买个收音机,就能收到消息
  • Android
    • 系统在运行过程中,会产生很多事件:开机、拨打电话、收发短信、电量改变、屏幕解锁
    • 一旦发生这种事件,系统会发送一个广播,为了接收到这个广播,知道系统发生了什么事件,就需要广播接收者,功能类似收音机

IP拨号器

功能:拨打电话时,自动添加ip线路的号码前缀

原理:系统打电话时,会发送一个打电话广播,iP拨号器只要定义广播接收者接收这个广播,那么就可以知道用户什么时候打电话,打电话广播中还会包含用户拨打的号码,ip拨号器可以从广播中获取此号码,并且作出修改,也就是添加号码前缀

短信拦截器

功能:类似短信防火墙系统在收到短信时,会产生一条短信广播,短信广播里,包含了短信的发信人号码和短信的内容,短信应用之所以能收到短信,其实是收到了短信广播,那我们的短信拦截器,只要在短信应用拿到广播之前,把短信广播拦截,那么短信应用就不会收到广播

监听sd卡状态

sd卡状态改变时,系统会发送广播

应用安装、更新、卸载的监听

原理,系统在安装更新卸载时,系统都会产生广播,广播中带有应用的包名

两种广播

无序广播

  • 所有intent-filter与无序广播匹配的广播接收者都可以收到这条广播,并且没有先后顺序(可以理解为同时)

有序广播

  • 所有intent-filter与有序广播匹配的广播接收者都可以收到这条广播,但是有先后顺序,按照intent-filter中定义优先级来决定(-1000~1000)

服务

  • 默默运行在后台的组件,可以理解为没有前台的Activity,服务的作用就是用来运行需要在后台默默运行的代码

进程优先级

  1. 前台进程:拥有一个可以与用户交互的Activity(Activity的onResume方法调用)的进程
  2. 可见进程:拥有一个可见但无焦点的Activity(Activity的onPause方法被调用)的进程
  3. 服务进程:拥有一个通过startService方法启动的服务的进程
  4. 后台进程:拥有一个不可见Activity(Activity的onStop方法被调用)的进程
  5. 空进程:!拥有任何活动的组件(Activity、service)的进程

电话窃.听器

用户接到来电,一旦接听,立即开始录音,电话挂掉,录音结束,生成音频文件

  • 电话的状态
    • 空闲
    • 响铃
    • 摘机 


服务两种启动方式

  • startService:服务被启动之后,跟启动它的组件没有一毛钱关系
  • bindService:跟启动它的组件同生共死
  • 绑定服务和解绑服务的生命周期方法:onCreate->onBind->onUnbind->onDestroy

找领导办证

  • 把服务看成一个领导,服务中有一个banZheng方法,如何才能访问?
  • 绑定服务时,会触发服务的onBind方法,此方法会返回一个Ibinder的对象给MainActivity,通过这个对象访问服务中的方法
  • 绑定服务

    Intent intent = new Intent(this, BanZhengService.class);
    bindService(intent, conn, BIND_AUTO_CREATE);
    
  • 绑定服务时要求传入一个ServiceConnection实现类的对象
  • 定义这个实现类

    class MyServiceconn implements ServiceConnection{
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
        zjr = (PublicBusiness) service;
    }
    @Override
    public void onServiceDisconnected(ComponentName name) { 
    }
    

    }

  • 创建实现类对象

     conn = new MyServiceconn();
    
  • 在服务中定义一个类实现Ibinder接口,以在onBind方法中返回

    class ZhongJianRen extends Binder implements PublicBusiness{
    public void QianXian(){
        //访问服务中的banZheng方法
        BanZheng();
    }   
    public void daMaJiang(){
    
    }
    

    }

  • 把QianXian方法抽取到接口PublicBusiness中定义

两种启动方法混合使用

  • 用服务实现音乐播放时,因为音乐播放必须运行在服务进程中,可是音乐服务中的方法,需要被前台Activity所调用,所以需要混合启动音乐服务
  • 先start,再bind,销毁时先unbind,在stop

使用服务注册广播接收者

  • Android四大组件都要在清单文件中注册
  • 广播接收者比较特殊,既可以在清单文件中注册,也可以直接使用代码注册
  • 有的广播接收者,必须代码注册

    • 电量改变
    • 屏幕锁屏和解锁
  • 注册广播接收者

    //创建广播接收者对象
    receiver = new ScreenOnOffReceiver();
    //通过IntentFilter对象指定广播接收者接收什么类型的广播
    IntentFilter filter = new IntentFilter();
    filter.addAction(Intent.ACTION_SCREEN_OFF);
    filter.addAction(Intent.ACTION_SCREEN_ON);
    
    //注册广播接收者
    registerReceiver(receiver, filter);
    
  • 解除注册广播接收者

    unregisterReceiver(receiver);
    
  • 解除注册之后,广播接收者将失去作用

本地服务:服务和启动它的组件在同一个进程

远程服务:服务和启动它的组件不在同一个进程

  • 远程服务只能隐式启动,类似隐式启动Activity,在清单文件中配置Service标签时,必须配置intent-filter子节点,并指定action子节点

AIDL

  • Android interface definition language
  • 安卓接口定义语言
  • 作用:跨进程通信
  • 应用场景:远程服务中的中间人对象,其他应用是拿不到的,那么在通过绑定服务获取中间人对象时,就无法强制转换,使用aidl,就可以在其他应用中拿到中间人类所实现的接口

支付宝远程服务

  1. 定义支付宝的服务,在服务中定义pay方法
  2. 定义中间人对象,把pay方法抽取成接口
  3. 把抽取出来的接口后缀名改成aidl
  4. 中间人对象直接继承Stub对象
  5. 注册这个支付宝服务,定义它的intent-Filter

需要支付的应用

  1. 把刚才定义好的aidl文件拷贝过来,注意aidl文件所在的包名必须跟原包名一致
  2. 远程绑定支付宝的服务,通过onServiceConnected方法我们可以拿到中间人对象
  3. 把中间人对象通过Stub.asInterface方法强转成定义了pay方法的接口
  4. 调用中间人的pay方法

五种前台进程

  1. activity执行了onresume方法,获得焦点
  2. 拥有一个跟正在与用户交互的activity绑定的服务
  3. 拥有一个服务执行了startForeground()方法
  4. 拥有一个正在执行onCreate()、onStart()或者onDestroy()方法中的任意一个的服务
  5. 拥有一个正在执行onReceive方法的广播接收者

两种可见进程

  1. activity执行了onPause方法,失去焦点,但是可见
  2. 拥有一个跟可见或前台activity绑定的服务

随堂笔记

服务的两种启动方式

startService

bindService

  • 此方法启动的服务,不会使进程称为服务进程,在服务运行列表根本找不到
  • 绑定服务启动的服务会和启动它的activity同生共死

服务的混合开启

  • start-bind-unbind-stop

在服务中用代码注册广播接收者

  • 广播接收者的使用必须在清单文件中注册
  • 其实也可以用代码注册
  • 但是基本用清单文件注册
  • 代码注册的好处就是,可以解除注册
  • 有些特殊广播接收者,只能代码注册
    • 电量改变
    • 屏幕解锁和锁屏

服务种类

本地服务

  • 启动的服务跟activity在同一个进程

远程服务

  • 启动的服务跟activity在不同的进程

AIDL

  • Android Interface definition Language
  • 安卓接口定义语言
  • 作用:进程间通信

进程优先级的补充

  • 前台进程:
    1. 拥有一个和用户正在交互的Activity绑定的服务的进程
    2. 拥有一个调用过startForeground方法的服务的进程
    3. 拥有一个正在执行生命周期方法(仅限onCreate、onStart、onDestroy)的服务的进程
    4. 拥有一个正在执行onReceive方法的广播接收者
  • 可见进程:拥有一个和可见Activity绑定的服务的进程 


服务两种启动方式

  • startService:服务被启动之后,跟启动它的组件没有一毛钱关系
  • bindService:跟启动它的组件同生共死
  • 绑定服务和解绑服务的生命周期方法:onCreate->onBind->onUnbind->onDestroy

找领导办证

  • 把服务看成一个领导,服务中有一个banZheng方法,如何才能访问?
  • 绑定服务时,会触发服务的onBind方法,此方法会返回一个Ibinder的对象给MainActivity,通过这个对象访问服务中的方法
  • 绑定服务

    Intent intent = new Intent(this, BanZhengService.class);
    bindService(intent, conn, BIND_AUTO_CREATE);
    
  • 绑定服务时要求传入一个ServiceConnection实现类的对象
  • 定义这个实现类

    class MyServiceconn implements ServiceConnection{
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
        zjr = (PublicBusiness) service;
    }
    @Override
    public void onServiceDisconnected(ComponentName name) { 
    }
    

    }

  • 创建实现类对象

     conn = new MyServiceconn();
    
  • 在服务中定义一个类实现Ibinder接口,以在onBind方法中返回

    class ZhongJianRen extends Binder implements PublicBusiness{
    public void QianXian(){
        //访问服务中的banZheng方法
        BanZheng();
    }   
    public void daMaJiang(){
    
    }
    

    }

  • 把QianXian方法抽取到接口PublicBusiness中定义

两种启动方法混合使用

  • 用服务实现音乐播放时,因为音乐播放必须运行在服务进程中,可是音乐服务中的方法,需要被前台Activity所调用,所以需要混合启动音乐服务
  • 先start,再bind,销毁时先unbind,在stop

使用服务注册广播接收者

  • Android四大组件都要在清单文件中注册
  • 广播接收者比较特殊,既可以在清单文件中注册,也可以直接使用代码注册
  • 有的广播接收者,必须代码注册

    • 电量改变
    • 屏幕锁屏和解锁
  • 注册广播接收者

    //创建广播接收者对象
    receiver = new ScreenOnOffReceiver();
    //通过IntentFilter对象指定广播接收者接收什么类型的广播
    IntentFilter filter = new IntentFilter();
    filter.addAction(Intent.ACTION_SCREEN_OFF);
    filter.addAction(Intent.ACTION_SCREEN_ON);
    
    //注册广播接收者
    registerReceiver(receiver, filter);
    
  • 解除注册广播接收者

    unregisterReceiver(receiver);
    
  • 解除注册之后,广播接收者将失去作用

本地服务:服务和启动它的组件在同一个进程

远程服务:服务和启动它的组件不在同一个进程

  • 远程服务只能隐式启动,类似隐式启动Activity,在清单文件中配置Service标签时,必须配置intent-filter子节点,并指定action子节点

AIDL

  • Android interface definition language
  • 安卓接口定义语言
  • 作用:跨进程通信
  • 应用场景:远程服务中的中间人对象,其他应用是拿不到的,那么在通过绑定服务获取中间人对象时,就无法强制转换,使用aidl,就可以在其他应用中拿到中间人类所实现的接口

支付宝远程服务

  1. 定义支付宝的服务,在服务中定义pay方法
  2. 定义中间人对象,把pay方法抽取成接口
  3. 把抽取出来的接口后缀名改成aidl
  4. 中间人对象直接继承Stub对象
  5. 注册这个支付宝服务,定义它的intent-Filter

需要支付的应用

  1. 把刚才定义好的aidl文件拷贝过来,注意aidl文件所在的包名必须跟原包名一致
  2. 远程绑定支付宝的服务,通过onServiceConnected方法我们可以拿到中间人对象
  3. 把中间人对象通过Stub.asInterface方法强转成定义了pay方法的接口
  4. 调用中间人的pay方法

五种前台进程

  1. activity执行了onresume方法,获得焦点
  2. 拥有一个跟正在与用户交互的activity绑定的服务
  3. 拥有一个服务执行了startForeground()方法
  4. 拥有一个正在执行onCreate()、onStart()或者onDestroy()方法中的任意一个的服务
  5. 拥有一个正在执行onReceive方法的广播接收者

两种可见进程

  1. activity执行了onPause方法,失去焦点,但是可见
  2. 拥有一个跟可见或前台activity绑定的服务

随堂笔记

服务的两种启动方式

startService

bindService

  • 此方法启动的服务,不会使进程称为服务进程,在服务运行列表根本找不到
  • 绑定服务启动的服务会和启动它的activity同生共死

服务的混合开启

  • start-bind-unbind-stop

在服务中用代码注册广播接收者

  • 广播接收者的使用必须在清单文件中注册
  • 其实也可以用代码注册
  • 但是基本用清单文件注册
  • 代码注册的好处就是,可以解除注册
  • 有些特殊广播接收者,只能代码注册
    • 电量改变
    • 屏幕解锁和锁屏

服务种类

本地服务

  • 启动的服务跟activity在同一个进程

远程服务

  • 启动的服务跟activity在不同的进程

AIDL

  • Android Interface definition Language
  • 安卓接口定义语言
  • 作用:进程间通信

进程优先级的补充

  • 前台进程:
    1. 拥有一个和用户正在交互的Activity绑定的服务的进程
    2. 拥有一个调用过startForeground方法的服务的进程
    3. 拥有一个正在执行生命周期方法(仅限onCreate、onStart、onDestroy)的服务的进程
    4. 拥有一个正在执行onReceive方法的广播接收者
  • 可见进程:拥有一个和可见Activity绑定的服务的进程 

服务两种启动方式

  • startService:服务被启动之后,跟启动它的组件没有一毛钱关系
  • bindService:跟启动它的组件同生共死
  • 绑定服务和解绑服务的生命周期方法:onCreate->onBind->onUnbind->onDestroy

找领导办证

  • 把服务看成一个领导,服务中有一个banZheng方法,如何才能访问?
  • 绑定服务时,会触发服务的onBind方法,此方法会返回一个Ibinder的对象给MainActivity,通过这个对象访问服务中的方法
  • 绑定服务

    Intent intent = new Intent(this, BanZhengService.class);
    bindService(intent, conn, BIND_AUTO_CREATE);
    
  • 绑定服务时要求传入一个ServiceConnection实现类的对象
  • 定义这个实现类

    class MyServiceconn implements ServiceConnection{
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
        zjr = (PublicBusiness) service;
    }
    @Override
    public void onServiceDisconnected(ComponentName name) { 
    }
    

    }

  • 创建实现类对象

     conn = new MyServiceconn();
    
  • 在服务中定义一个类实现Ibinder接口,以在onBind方法中返回

    class ZhongJianRen extends Binder implements PublicBusiness{
    public void QianXian(){
        //访问服务中的banZheng方法
        BanZheng();
    }   
    public void daMaJiang(){
    
    }
    

    }

  • 把QianXian方法抽取到接口PublicBusiness中定义

两种启动方法混合使用

  • 用服务实现音乐播放时,因为音乐播放必须运行在服务进程中,可是音乐服务中的方法,需要被前台Activity所调用,所以需要混合启动音乐服务
  • 先start,再bind,销毁时先unbind,在stop

使用服务注册广播接收者

  • Android四大组件都要在清单文件中注册
  • 广播接收者比较特殊,既可以在清单文件中注册,也可以直接使用代码注册
  • 有的广播接收者,必须代码注册

    • 电量改变
    • 屏幕锁屏和解锁
  • 注册广播接收者

    //创建广播接收者对象
    receiver = new ScreenOnOffReceiver();
    //通过IntentFilter对象指定广播接收者接收什么类型的广播
    IntentFilter filter = new IntentFilter();
    filter.addAction(Intent.ACTION_SCREEN_OFF);
    filter.addAction(Intent.ACTION_SCREEN_ON);
    
    //注册广播接收者
    registerReceiver(receiver, filter);
    
  • 解除注册广播接收者

    unregisterReceiver(receiver);
    
  • 解除注册之后,广播接收者将失去作用

本地服务:服务和启动它的组件在同一个进程

远程服务:服务和启动它的组件不在同一个进程

  • 远程服务只能隐式启动,类似隐式启动Activity,在清单文件中配置Service标签时,必须配置intent-filter子节点,并指定action子节点

AIDL

  • Android interface definition language
  • 安卓接口定义语言
  • 作用:跨进程通信
  • 应用场景:远程服务中的中间人对象,其他应用是拿不到的,那么在通过绑定服务获取中间人对象时,就无法强制转换,使用aidl,就可以在其他应用中拿到中间人类所实现的接口

支付宝远程服务

  1. 定义支付宝的服务,在服务中定义pay方法
  2. 定义中间人对象,把pay方法抽取成接口
  3. 把抽取出来的接口后缀名改成aidl
  4. 中间人对象直接继承Stub对象
  5. 注册这个支付宝服务,定义它的intent-Filter

需要支付的应用

  1. 把刚才定义好的aidl文件拷贝过来,注意aidl文件所在的包名必须跟原包名一致
  2. 远程绑定支付宝的服务,通过onServiceConnected方法我们可以拿到中间人对象
  3. 把中间人对象通过Stub.asInterface方法强转成定义了pay方法的接口
  4. 调用中间人的pay方法

五种前台进程

  1. activity执行了onresume方法,获得焦点
  2. 拥有一个跟正在与用户交互的activity绑定的服务
  3. 拥有一个服务执行了startForeground()方法
  4. 拥有一个正在执行onCreate()、onStart()或者onDestroy()方法中的任意一个的服务
  5. 拥有一个正在执行onReceive方法的广播接收者

两种可见进程

  1. activity执行了onPause方法,失去焦点,但是可见
  2. 拥有一个跟可见或前台activity绑定的服务

随堂笔记

服务的两种启动方式

startService

bindService

  • 此方法启动的服务,不会使进程称为服务进程,在服务运行列表根本找不到
  • 绑定服务启动的服务会和启动它的activity同生共死

服务的混合开启

  • start-bind-unbind-stop

在服务中用代码注册广播接收者

  • 广播接收者的使用必须在清单文件中注册
  • 其实也可以用代码注册
  • 但是基本用清单文件注册
  • 代码注册的好处就是,可以解除注册
  • 有些特殊广播接收者,只能代码注册
    • 电量改变
    • 屏幕解锁和锁屏

服务种类

本地服务

  • 启动的服务跟activity在同一个进程

远程服务

  • 启动的服务跟activity在不同的进程

AIDL

  • Android Interface definition Language
  • 安卓接口定义语言
  • 作用:进程间通信

进程优先级的补充

  • 前台进程:
    1. 拥有一个和用户正在交互的Activity绑定的服务的进程
    2. 拥有一个调用过startForeground方法的服务的进程
    3. 拥有一个正在执行生命周期方法(仅限onCreate、onStart、onDestroy)的服务的进程
    4. 拥有一个正在执行onReceive方法的广播接收者
  • 可见进程:拥有一个和可见Activity绑定的服务的进程 

你可能感兴趣的:(Android应用开发基础之广播与服务)