安卓四大组件之--广播接受者

本文深入探讨Android系统中的广播机制,包括广播的概念、广播接收者的作用、IP拨号器的实现、短信拦截器的原理、监听SD卡状态的方法、勒索软件的触发方式、应用安装、卸载、更新的监听以及广播的类型(无序广播与有序广播)。同时,介绍了Service组件的工作流程、进程优先级分类、电话状态监听的实现,旨在全面掌握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.itheima.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.itheima.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.itheima.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.itheima.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.itheima.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. 服务进程:拥有一个通过startService方法启动的服务的进程
  4. 后台进程:拥有一个后台activity(onStop方法被调用)的进程
  5. 空进程:没有拥有任何活动的应用组件的进程,也就是没有任何服务和activity在运行

电话窃听器

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

    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四大组件的关系与交互 #### 1. **Activity** Activity 是 Android 应用程序的核心组件之一,用于展示用户界面并与用户进行交互。它通常作为应用程序的主要入口点[^2]。 - Activity 可以启动其他 Activity 或 Service 来完成特定的任务。 - 它可以通过 `Intent` 启动 Broadcast Receiver 并发送广播消息[^4]。 - 如果需要与其他应用共享数据,则可以借助 Content Provider 实现跨进程通信。 ```java // 示例:通过 Intent 启动另一个 Activity Intent intent = new Intent(CurrentActivity.this, TargetActivity.class); startActivity(intent); // 示例:通过 Intent 发送广播 Intent broadcastIntent = new Intent("com.example.CUSTOM_BROADCAST"); sendBroadcast(broadcastIntent); ``` --- #### 2. **Service** Service 负责执行长时间运行的操作,比如播放音乐、下载文件等背景任务。它可以独立于 UI 存在[^3]。 - Service 可被 Activity 启动或绑定到某个 Activity 上工作。 - 当有后台操作完成后,可通过广播通知 Activity 更新其状态。 - 对于某些场景下,可能还需要访问数据库或其他资源时,可利用 Content Provider 提供统一接口管理这些资源。 ```java // 示例:从 Activity 中启动 Service Intent serviceIntent = new Intent(this, MyBackgroundService.class); startService(serviceIntent); // 示例:停止 Service stopService(new Intent(this, MyBackgroundService.class)); ``` --- #### 3. **BroadcastReceiver** BroadcastReceiver 主要用来监听来自系统或者其他 App 的全局事件,并作出响应处理。 - 可由任何地发出广播信号激活(包括但不限于 Activity 和 Service)。 - 接受到的消息能够进一步影响当前正在显示的 Activity 行为或者触发新的业务逻辑流程。 当涉及到敏感权限控制的应用间通讯时,建议采用 LocalBroadcastManager 替代传统法减少安全隐患同时提高效率[^1]。 ```java // 注册动态广播接收器 registerReceiver(myDynamicReceiver, new IntentFilter("ACTION_NAME")); // 解除注册 unregisterReceiver(myDynamicReceiver); ``` --- #### 4. **ContentProvider** ContentProvider 提供了一种机制让不同 Apps 之间安全地交换信息[^3]。 - 其他三个组件都可以成为它的客户端去请求读写指定 URI 下代表的数据集。 - 假设某项功能需求涉及多处异步加载远程服务器上的资料列表再呈现给最终使用者看的话,那么很可能会设计成先经由网络层获取原始 JSON/XML 数据交给本地 SQLite 缓存起来并通过自定义实现好的 ContentResolver 查询出来供给前端渲染使用。 ```java // 访问 ContentProvider 中的内容 Uri contentUri = Uri.parse("content://example.provider/data"); Cursor cursor = getContentResolver().query(contentUri, null, null, null, null); if (cursor != null && cursor.moveToFirst()) { do { String value = cursor.getString(cursor.getColumnIndexOrThrow("column_name")); Log.d("Data", "Value: " + value); } while (cursor.moveToNext()); } if (cursor != null) { cursor.close(); } ``` --- ### 组件间的协作模式总结表 | **发起** | **目标对象** | **典型应用场景** | |------------------|--------------------|----------------------------------------------------------------------------| | Activity | Another Activity | 页面跳转 | | | Service | 开启后台任务 | | | BroadcastReceiver | 发布广播 | | | ContentProvider | 请求外部数据源 | | Service | Activity | 返回结果更新UI | | | BroadcastReceiver | 结束后发布广播告知已完成 | | BroadcastReceiver | Activity/Service | 收到广播后启动相应组件 | --- 相关问题
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值