【Android Broadcast】BroadcastReceiver

作为安卓中的四大组件广播的作用也是非常大的,听名字我们或许就知道他的功能啦!接下来就回顾、总结下这个重要的组件。

在这里插入图片描述

一、简介

1、分类
  • 广播发送者:发送广播,发布广播后,订阅过的接收者就可以接收到广播。
  • 广播接收者:接收广播(想要接收广播,必须要注册即订阅相应的广播)
2、BroadcastReceiver定义

BroadcastReceiver是一个全局的监听器,它可以监听安卓系统、其他App、自己App发出的广播。并作出相应的处理。是安卓的四大组件之一。

3、BroadcastReceiver的作用

BroadcastReceiver就是广播接收者,接收广播。

4、应用场景

(1)同一app内部的同一组件内的消息通信(单个或多个线程之间)
(2)同一app内部的不同组件之间的消息通信(单个进程)
(3)同一app有不同的组件,且运行在不同的进程中(多个进程之间)
(4)不同app之间的组件之间消息通信(多个进程之间)
(5)Android系统在特定情况下与App之间的消息通信

  • 场景1实际上没必要用广播实现。通过扩展作用于的范围、使用接口回调、使用handler机制都可以解决
  • 场景2同一app内部的不同组件之间的消息通信(单个进程),对于此类需求,在有些教复杂的情况下单纯的依靠基于接口的回调等方式不好处理,此时可以直接使用EventBus等,相对而言,EventBus由于是针对统一进程,用于处理此类需求非常适合,且轻松解耦。
  • 场景3、4、5可以使用广播
5、原理基于

发布者、订阅者,一听我们就想到了“观察者设计模式”,没错安卓中的广播机制就是基于观察者模式的,只不过安卓中通过“消息中心ASM”来处理沟通广播接收者、发送者。为啥要把ASM设计为消息中心呢?

  • 这就考虑到广播发送者、接收者的作用了即发送、接收全局消息。这时肯定不局限与同一个app内的进程中啦。而安卓的ipc机制底层又是基于Binder的所以广播发送者、广播接收者、ASM之间就通过Binder联系起来啦!
  • 由于四大组件的启动都牵涉到ASM所以,我们启动广播也是离不开他的,ASM便可根据发送者的要求在安卓的注册列表中寻找合适的广播(寻找依据: 广播的权限、intent filter 内的匹配规则)

(1)原理草图

在这里插入图片描述

1、广播接收者BroadcastReceiver通过Binder机制向AMS(Activity Manager Service)进行注册;
2、广播发送者通过binder机制向AMS发送广播;
3、AMS查找符合相应条件(IntentFilter/Permission等)的BroadcastReceiver,将广播发送到BroadcastReceiver(一般情况下是Activity)相应的消息循环队列中;
4、消息循环执行拿到此广播,回调BroadcastReceiver中的onReceive()方法。

二、使用流程

1、创建广播接收者

步骤:
1、自定义类继承BroadcastReceiver
2、重写onReceiver方法
3、等待广播发送者发送广播,这样BroadcastReceiver的onReceiver就会回调。

/**
 * Created by sunnyDay on 2019/8/13 18:09
 */
public class MyBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
      Toast.makeText(context, "收到广播", Toast.LENGTH_SHORT).show();
    }
}

2、广播的注册

(1)动态注册

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onResume() {
        super.onResume();

        IntentFilter filter = new IntentFilter();
        filter.addAction("sunny.day.custom.broadcast"); //添加过滤条件,接收广播
        MyBroadcastReceiver mBroadcastReceiver = new MyBroadcastReceiver();
        registerReceiver(mBroadcastReceiver, filter);
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    /**
     * 点击按钮发送自定义广播
     * */
    public void send(View view) {
        Intent intent = new Intent();
        intent.setAction("sunny.day.custom.broadcast");// 发送者定义 过滤条件
        sendBroadcast(intent);
    }
    
    @Override
    protected void onPause() {
        super.onPause();
        // activity 销毁时解注册
        unregisterReceiver(mBroadcastReceiver);
    }
}

说明:

  • 如上通过sendBroadcast 就可发送一条自定义广播。
  • 通过代码中registerReceiver就可注册想要接收的广播。当有广播发送时,广播接收者就会收到。
  • 上面的action是我们自己定义的字符串、其实安卓系统也定义了很多,比如“android.net.conn.CONNECTIVITY_CHANGE”网络状态变化。我们也可以监听系统广播。监听系统广播时只需要注册下系统广播即可。当系统发送广播时,我们就可以收到。

注意:

  • 动态注册的广播要在activity退出时解注册,否则会造成内存泄漏。
  • 广播接收器也是运行在UI线程,因此,onReceive方法中不能执行太耗时的操作。否则将因此ANR。一般情况下,根据实际业务需求,onReceive方法中都会涉及到与其他组件之间的交互,如发送Notification、启动service等。

广播解注册的时机:

不在onCreate() & onDestory() 或 onStart() & onStop()注册、注销是因为:

当系统因为内存不足(优先级更高的应用需要内存)要回收Activity占用的资源时,Activity在执行完onPause()方法后就会被销毁,有些生命周期方法onStop(),onDestory()就不会执行。当再回到此Activity时,是从onCreate方法开始执行。

假设我们将广播的注销放在onStop(),onDestory()方法里的话,有可能在Activity被销毁后还未执行onStop(),onDestory()方法,即广播仍还未注销,从而导致内存泄露。但是,onPause()一定会被执行,从而保证了广播在App死亡前一定会被注销,从而防止内存泄露。

(2)静态注册

在安卓的manifest中通过receiver节点下添加 intent-filter,然后在intent-filter中添加过滤条件。

        <receiver android:name=".MyBroadcastReceiver"
            android:exported="false">
            <intent-filter >
                // 自定义一个action
                <action android:name="sunny.day.custom.broadcast" />
            </intent-filter>
        </receiver>

receiver节点的属性:
1、exported :此BroadcastReceiver能否接收其他App的发出的广播,这个属性默认值有点意思,其默认值是由receiver中有无intent-filter决定的,如果有intent-filter,默认值为true,否则为false。(同样的,activity/service中的此属性默认值一样遵循此规则)同时,需要注意的是,这个值的设定是以application或者application user id为界的,而非进程为界(一个应用中可能含有多个进程)。
2、name :此BroadcastReceiver类名。
3、permission:如果设置,具有相应权限的广播发送方发送的广播才能被此broadcastReceiver所接收。
4、process:四大组件都具有这个属性。为组件指定单独的进程。不指定默认为app默认的进程。

    /**
     * 点击按钮发送自定义广播(隐式发送)
     */
   private fun send1() {
        val intent = Intent()
        intent.action = "sunny.day.custom.broadcast"
        sendBroadcast(intent)
    }

    /**
     * 点击按钮发送自定义广播(显式发送)
     */
    private fun send2() {
        val intent = Intent(this,MyBroadcastReceiver::class.java)
        intent.action = "sunny.day.custom.broadcast"
        sendBroadcast(intent)
    }

👆🏻两种方法发送广播,需要注意的是隐式方式发送广播时,静态注册的广播接受者收不到广播的。如上send1发送广播时MyBroadcastReceiver#onReceive是不会回调的。

虽然静态注册的自定义广播,隐式启动时完全失效。但是我们可以显式启动啊,如上使用send2发送的广播MyBroadcastReceiver#onReceive是可以收到回调的。

安卓8.0开始限制了静态广播的注册建议:如果您的应用以 API 级别 26 或更高级别的平台版本为目标,则不能使用清单为隐式广播(没有明确针对您的应用的广播)声明接收器,但一些不受此限制的隐式广播除外。在大多数情况下,您可以使用调度作业来代替。

因此我们可以知道:

  • 针对系统发出的广播:我们要了解清楚这个广播是否是不受限制的广播,否则就要动态注册接收。
  • 针对自定义广播:建议也是动态注册,要是静态注册时则需要发送广播时显式发送。

(3)动态静态注册的区别

1、静态注册:

优点:属于常驻广播,清单文件注册后不受任何组件的影响。即使我们的app退出关闭后有广播来了我们的app依旧会被系统调用。
缺点:由于需要时刻监听广播所以耗电、占内存。

2、动态注册:

特点:灵活,跟随组件的生命周期变化。组件销毁时必须要解注册广播,否则容易造成内存泄漏。

三、广播的分类

1、普通广播

普通广播就是我们简单自定义的广播。通过intent携带actionActivity组件发送一个广播,想要接收这个普通广播只需给相应的广播接收者注册下即可:

// 发送普通广播
 public void send(View view) {
        Intent intent = new Intent();
        intent.setAction("sunny.day.custom.broadcast");
        sendBroadcast(intent);
    }
// 广播接收者注册广播
   IntentFilter filter = new IntentFilter();
        filter.addAction("sunny.day.custom.broadcast"); //添加过滤条件,接收广播
        MyBroadcastReceiver mBroadcastReceiver = new MyBroadcastReceiver();
        registerReceiver(mBroadcastReceiver, filter);

(2)广播的权限添加

有些广播为了安全或许会添加一些自定义权限,如下我们在清单文件中自定义一个权限:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
          package="com.sunnyday.broadcastreceiverrevision">
          
  <!-- 自定义权限-->
    <permission
            android:name="this.is.custom.permission"

            android:protectionLevel="signature"/>
</manifest>

注册广播添加权限:

IntentFilter filter = new IntentFilter();
        filter.addAction("sunny.day.custom.broadcast"); //添加过滤条件,接收广播
        mBroadcastReceiver = new MyBroadcastReceiver();
        registerReceiver(mBroadcastReceiver, filter,"this.is.custom.permission",null);

这时我们点击按钮会发现没收到广播。。。。。原因是你的应用程序没有这个"this.is.custom.permission"权限。你需要声明下。和平常一样: 清单文件<uses-permission android:name=“this.is.custom.permission”/>申请下即可。

2、系统广播

安卓系统定义个好多广播:比如网络变化会发出相应的广播、来电收到短信也会收到相应的广播、手机开关机、电量低等等、、、、、、

(1)使用

我们不需要手动发送这些广播,这些广播是系统再特定的时机发送的。我们只需注册下相应的action即可。

(2)常见的系统广播action

在这里插入图片描述

3、有序广播

前面我们接收系统的广播,接收自己自定义的广播都是无序广播,只要订阅了发送者的广播,所有接收者都可以同时接收到。没有先后顺序。接下来便看下有序广播。

(1)接收顺序

1、发出广播后,接收者按照优先级顺序接收广播。
2、所以有序广播是针对广播接收者而言的。

(2)广播接收者接收广播的规则

1、按照Priority属相的大小顺序(大的优先级大)
2、代码注册优先于静态注册(同优先级下)
ps:Priority为intent filter的属性,可以代码设置也可以清单文件属性添加。

(3)广播特点

1、广播接收者按照顺序接收广播(优先级)
2、优先级高的接收者接收广播后可以拦截广播,让优先级低的接收不到。
3、优先级高的广播接收者还可以对广播的内容进行修改,这样优先级低的收到的就是修改后的广播内容。

(4)发送

sendOrderedBroadcast(intent,null)
4、粘性广播

粘性广播,在 android 5.0/api 21中deprecated,不再推荐使用,相应的还有粘性有序广播,同样已经deprecated。

5、本地广播

(1)为啥使用app内部广播

Android中的广播可以跨App直接通信(exported对于有intent-filter情况下默认值为true)这会导致安全问题:
1、其他app针对我们广播接收器匹配规则发出相应的action的广播,我们的接收器就接收、处理。
2、其他app注册了与app一致的action便可获取广播信息

(2)安全问题的解决

1、注册广播时将exported属性设置为false,使得非本App内部发出的此广播不被接收;
2、在广播发送和接收时,增设相应权限permission,用于权限验证;
3、发送广播时指定该广播接收器所在的包名,此广播将只会发送到此包中的App内与之相匹配的有效广播接收器中。(intent.setPackage(packageName))
安卓针对上面的解决思路进行了封装:LocalBroadcastManager类

(3)使用LocalBroadcastManager发送应用内的广播

LocalBroadcastManager这种发送方式只能代码注册。


public class MainActivity extends AppCompatActivity {

    private MyBroadcastReceiver mBroadcastReceiver;
    private static LocalBroadcastManager localBroadcastManager; // 获得单例

    @Override
    protected void onResume() {
        super.onResume();
        localBroadcastManager = LocalBroadcastManager.getInstance(this);

        IntentFilter filter = new IntentFilter();
        filter.addAction("sunny.day.custom.broadcast"); //添加过滤条件,接收广播
        mBroadcastReceiver = new MyBroadcastReceiver();

        localBroadcastManager.registerReceiver(mBroadcastReceiver, filter); //注册广播
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

    }

    /**
     * 点击按钮发送自定义 app内部广播
     */
    public void send(View view) {
        Intent intent = new Intent();
        intent.setAction("sunny.day.custom.broadcast");
        localBroadcastManager.sendBroadcast(intent);  //发送应用内广播
    }

    @Override
    protected void onPause() {
        super.onPause();
        // activity 销毁时解注册
        localBroadcastManager.unregisterReceiver(mBroadcastReceiver);
    }
}

和传统的自定义广播区别:
1、使用了LocalBroadcastManager 类来管理
2、通过LocalBroadcastManager 类发送广播
3、通过LocalBroadcastManager 注册送广播
4、通过LocalBroadcastManager 解绑送广播

end

参考:
android:exported 属性详解

官方文档

  • 0
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值