Android基础知识巩固系列 Android之四大组件——BroadcastReceiver(广播)

因为最近要面试,于是打算整理整理一下Android的基础知识,由于之前本人已经学习过大概的Android基础知识,这里主要讲这四大组件、五大存储、六大布局、网络请求等这些内容,其他一些等有时间再整理,话不多说。

应用组件(官方解释,需科学上网)

应用组件是 Android 应用的基本构建基块。每个组件都是一个不同的点,系统可以通过它进入您的应用。 并非所有组件都是用户的实际入口点,有些组件相互依赖,但每个组件都以独立实体形式存在,并发挥特定作用 — 每个组件都是唯一的构建基块,有助于定义应用的总体行为。

共有四种不同的应用组件类型。每种类型都服务于不同的目的,并且具有定义组件的创建和销毁方式的不同生命周期。

一、什么是BoradcastReceiver(广播)?

官方的解释中可以看到:

BroadcastReceiver

BroadcastReceiver是一种用于响应系统范围广播通知的组件。 许多广播都是由系统发起的 —— 例如,通知屏幕已关闭、电池电量不足或已拍摄照片的广播。应用也可以发起广播 — 例如,通知其他应用某些数据已下载至设备,并且可供其使用。 尽管广播接收器不会显示用户界面,但它们可以创建状态栏通知,在发生广播事件时提醒用户。 但广播接收器更常见的用途只是作为通向其他组件的“通道”,设计用于执行极少量的工作。 例如,它可能会基于事件发起一项服务来执行某项工作。

参考网站: https://blog.csdn.net/shenggaofei/article/details/52450668?utm_source=blogxgwz3  https://www.jianshu.com/p/ca3d87a4cdf3

通俗一点地来说,你的应用可以使用它对外部事件进行过滤,只对感兴趣的外部事件(如当电话呼入时,或者数据网络可用时)进行接收并做出响应。广播接收器没有用户界面。然而,它们可以启动一个activity或serice来响应它们收到的信息,或者用NotificationManager来通知用户。通知可以用很多种方式来吸引用户的注意力,例如闪动背灯、震动、播放声音等。一般来说是在状态栏上放一个持久的图标,用户可以打开它并获取消息。

即BroadcastReceiver(广播)是一个全局的监听器,监听/接收应用APP发出的广播消息,并做出响应。

注:Android广播分为两个角色:广播发送者、广播接收者

二、BoradcastReceiver(广播)的应用场景。

参考网站:https://www.jianshu.com/p/8650b3878722

广播是一种机制,而且是双向的(工作线程可以往主线程发广播,主线程也可以往工作线程发广播),应用的场景有如下几点:

(1)同一APP内部的同一组件内的消息通信(单个或多个线程之间)

(2)同一APP内部的不同组件之间的消息通信(单个线程)

(3)P具同一AP有多个进程的不同组件之间的消息通信

(4)不同APP之间的组件之间消息通信

(5)Android系统在特定情况下与APP之间的消息通信(系统广播)如:电话呼入时、网络可用时

通俗来讲:第一种是系统广播,第二种是可以替换Handler用来作为线程间的通信桥梁

三、BoradcastReceiver(广播)的实现原理。

参考网站:https://www.jianshu.com/p/ca3d87a4cdf3

3.1 采用的模型

Android中的广播使用了设计模式中的观察者模式:基于消息的发布/订阅事件模型。

因此,Android将广播的发送者和接收者解耦,使得系统方便集成,更易拓展。

3.2 模型的讲解

模型中有三个角色:

1.消息订阅者(广播接受者)  2.消息发布者(广播发布者)  3.消息中心(AMS,即Activity Manager Service)

示意图&原理如下:

四、BoradcastReceiver(广播)的使用。

参考网站:https://www.jianshu.com/p/ca3d87a4cdf3

使用流程图:


4.1 自定义广播接收者BroadcastReceiver

首先继承BroadcastReceiver基类,然后重写抽象方法onReceive(),例如:

//继承BroadcastReceiver基类
public class MyBroadcastReceiver extends BroadcastReceiver {

    //重写onReceive方法  接收到广播后,则自动调用该方法
    @Override
    public void onReceive(Context context, Intent intent) {
        //写入接收广播后的操作或响应
        Toast.makeText(context, "网络发生了改变", Toast.LENGTH_SHORT).show();
    }
}

注:1.广播接收器接收到相应广播后,会自动回调onReceive()方法。

       2.一般情况下,onReceive方法会涉及与其他组件之间的交互,如发送Notification(通知)、启动Service(服务)等。

       3.默认情况下,广播接受者运行在UI线程,因此,onReceive()方法不能执行耗时的操作,否则将导致ANR(无响应)。

4.2 广播接收器注册

注册的方式分为两种:静态注册、动态注册


静态注册:

  • 注册方式:在AndroidManifest.xml里通过<receive>标签声明
  • 属性说明:
<receiver 
    android:enabled=["true" | "false"]
//此broadcastReceiver能否接收其他App的发出的广播
//默认值是由receiver中有无intent-filter决定的:如果有intent-filter,默认值为true,否则为false
    android:exported=["true" | "false"]
    android:icon="drawable resource"
    android:label="string resource"
//继承BroadcastReceiver子类的类名
    android:name=".mBroadcastReceiver"
//具有相应权限的广播发送者发送的广播才能被此BroadcastReceiver所接收;
    android:permission="string"
//BroadcastReceiver运行所处的进程
//默认为app的进程,可以指定独立的进程
//注:Android四大基本组件都可以通过此属性指定自己的独立进程
    android:process="string" >

//用于指定此广播接收器将接收的广播类型
//本示例中给出的是用于接收网络状态改变时发出的广播
  <intent-filter>
    <action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
  </intent-filter>
</receiver>
  • 代码示例:
            <receiver
                android:name=".MyBroadcastReceiver"
                android:enabled="true"
                android:exported="true">
                <intent-filter>
                    <!-- 用于接收网络状态改变时发出的广播 -->
                    <action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
                </intent-filter>
            </receiver>

    当此APP首次启动时,系统会自动实例化MyBroadcastReceiver类,并注册到系统中。


动态注册:

  • 注册方式:在代码中调用Context.registerReceiver()方法
  • 特别注意:动态广播最好在活动的onResume中注册、onPause中注销原因:

          1.对于动态广播,有注册就必然得有注销,否则会导致内存泄漏!重复注册注销也不允许

          2.在onResume注册、onPause注销是因为onPause在活动销毁之前一定会被执行,从而保证广播在活动销毁之前一定会被注               销,从而防止内存泄露。

              不在 onCreate & onDestroy 或者 onStart & onStop 注册注销是因为:

                (1)当系统因为内存不足要回收Activity占用的资源时,Activity在执行完onPause方法后就会被销毁,那么 onStop和                                 onDestroy方法就不会执行。当再回到此Activity时,是从onCreate开始执行的。(在某篇帖子下面的人评论说:如果                           写在onDestroy中unregister,当由于处于后台在内存紧张时杀掉而没有跑onDestroy()也没有关系,app进程都被杀掉                           了,之前注册的自然也没有了)

                (2)但是,onPause一定会被执行,从而保证了广播在活动销毁之前一定会被注销,从而防止内存泄漏。

  • 代码示例:
    @Override
    protected void onResume() {
        super.onResume();
        // 1. 实例化BroadcastReceiver子类 &  IntentFilter
        mNetworkChangeReceiver = new NetworkChangeReceiver();
        mIntentFilter = new IntentFilter();

        // 2. 设置接收广播的类型
        mIntentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");

        // 3. 动态注册:调用Context的registerReceiver()方法
        registerReceiver(mNetworkChangeReceiver, mIntentFilter);
    }

    // 注册广播后,要在相应位置记得销毁广播
    // 即在onPause() 中unregisterReceiver(mBroadcastReceiver)
    // 当此Activity实例化时,会动态将MyBroadcastReceiver注册到系统中
    // 当此Activity销毁时,动态注册的MyBroadcastReceiver将不再接收到相应的广播。
    @Override
    protected void onPause() {
        super.onPause();
        //销毁在onResume()方法中的广播
        unregisterReceiver(mNetworkChangeReceiver);
    }

    private class NetworkChangeReceiver extends BroadcastReceiver {

        @Override
        public void onReceive(Context context, Intent intent) {
            ConnectivityManager cm = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
            NetworkInfo networkInfo = cm.getActiveNetworkInfo();
            if (networkInfo != null && networkInfo.isAvailable()) {
                Toast.makeText(getApplicationContext(), "当前有网络!", Toast.LENGTH_SHORT).show();
            } else {
                Toast.makeText(getApplicationContext(), "当前无网络!", Toast.LENGTH_SHORT).show();
            }
        }
    }

3.两种注册方式的区别

五、BoradcastReceiver(广播)的使用(广播发送者向AMS发送广播)。

5.1 广播的发送

  • 广播是通过用Intent(意图)标识
  • 定义广播的本质 = 定义广播所具备的Intent(意图)
  • 广播发送 = 广播发送者 将此广播的Intent(意图)通过sendBroadcast()方法发送出去

5.2 广播的类型

  • 普通广播 || 标准广播(Normal Broadcast
  • 有序广播(Ordered Broadcast
  • 系统广播(System Broadcast
  • 本地广播 || App应用内广播(Local Broadcast
  • 粘性广播(Sticky Broadcast

1.普通广播 || 标准广播(Normal Broadcast

即开发者自身定义Intent的广播(最常用),发送广播的使用如下:

(1)先定义一个广播接收器

public class MyBroadcastReceiver extends BroadcastReceiver {

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

 (2)在manifest清单文件加入Intent-filter标签,在Intent-filter中加入action,指定广播接收者要接收action值为XXX的广播

        <receiver
            android:name=".MyBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="MY_BROADCAST" />
            </intent-filter>
        </receiver>

 (3)在Inten中指定action的值,然后通过sendBroadcast方法发送广播

public class MainActivity extends AppCompatActivity {

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

        Button button = findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent("MY_BROADCAST");
                sendBroadcast(intent);
            }
        });
    }
}

 注:若已注册的广播接收者中注册时IntentFilter的action值与指定的action值一致,则会接收到此广播(即进行回调onReceive方法)

2.有序广播(Ordered Broadcast

  • 定义

           发送出去的广播被广播接收者按照先后顺序接收(有序是针对广播接收者而言的)

  • 广播接收者接收广播的顺序规则(同时面向静态和动态注册的广播接收者)

           1.按照Priority属性值从大到小排序;

           2.Priority属性相同者,默认动态注册的广播优先;

  • 特点

           1.接收广播按顺序接收

           2.先接收的广播接收者可以对广播进行截断,即后接收的广播接收者不再接收到此广播;

           3.先接收的广播接收者可以对广播进行修改,即后接收的广播接收者将接收到修改后的广播;(这里应该是指修改Intent中的                数据)

  • 具体使用

          有序广播的使用与标准广播(普通广播)的使用非常类似,差异仅仅只在于广播的发送方式。

        (1)首先还是定义广播接收器,这里定义两个广播接收器,方便看出有序广播的特点

//第一个
public class MyBroadcastReceiver extends BroadcastReceiver {

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

//第二个
public class SecondBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "这是第二个广播!", Toast.LENGTH_SHORT).show();
    }
}

         (2)同样在manifest清单文件加入Intent-filter标签,在Intent-filter中加入action,指定广播接收者要接收action值为XXX的广                        播,在Intent-filter标签中可以通过priority来设置优先级的大小

        </receiver>
        <receiver
            android:name=".MyBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter android:priority="100">
                <action android:name="MY_BROADCAST" />
            </intent-filter>
        </receiver>
        <receiver
            android:name=".SecondBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter android:priority="99">
                <action android:name="MY_BROADCAST" />
            </intent-filter>

          (3)在Inten中指定action的值,然后通过sendOrderedBroadcast方法发送广播,第二个参数是权限相关字符串

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button button = findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent("MY_BROADCAST");
                sendOrderedBroadcast(intent,null);
            }
        });
    }
}

注:以上运行后,第一个弹出的是权优先级最高的 MyBroadcastReceiver,假设要截断广播则调用abortBroadcast方法,如下

public class MyBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "第一个广播收到广播消息!", Toast.LENGTH_SHORT).show();
        abortBroadcast(); //截断广播
    }
}

 截断后,第二个广播接收器再也接收不到广播,自然也就弹不出来吐司提示了

3.系统广播(System Broadcast

  • Android中内置了多个系统广播:只要涉及到手机的基本操作(如开机、网络状态变化、拍照等等),系统都会发出相应的广播。
  • 每个广播都有特定的Intent-Filter(包括具体的action值),Android系统广播action值对应表如下:
系统操作action
监听网络变化android.net.conn.CONNECTIVITY_CHANGE
关闭或打开飞行模式Intent.ACTION_AIRPLANE_MODE_CHANGED
充电时或电量发生变化Intent.ACTION_BATTERY_CHANGED
电池电量低Intent.ACTION_BATTERY_LOW
电池电量充足(即从电量低变化到饱满时会发出广播)Intent.ACTION_BATTERY_OKAY
系统启动完成后(仅广播一次)Intent.ACTION_BOOT_COMPLETED
按下照相时的拍照按键(硬件按键时)Intent.ACTION_CAMERA_BUTTON
屏幕锁屏Intent.ACTION_CLOSE_SYSTEM_DIALOGS
设备当前设置被改变时(界面语言、设备方向等)Intent.ACTION_CONFIGURATION_CHANGED
插入耳机时Intent.ACTION_HEADSET_PLUG
未正确移除SD卡但已取出来时(正确移除方法:设置-->SD卡和设备内存-->卸载SD卡)Intent.ACTION_MEDIA_BAD_REMOVAL
插入外部储存装置(如SD卡)Intent.ACTION_MEDIA_CHECKING
成功安装APKIntent.ACTION_PACKAGE_ADDED
成功删除APKIntent.ACTION_PACKAGE_REMOVED
重启设备Intent.ACTION_REBOOT
屏幕被关闭Intent.ACTION_SCREEN_OFF
屏幕被打开Intent.ACTION_SCREEN_ON
关闭系统时Intent.ACTION_SHUTDOWN

注:部分手机可能某些原因会屏蔽系统的广播,比如我之前看《第一行代码》的时候,发现小米会屏幕系统的开机广播。

4.本地广播 || App应用内广播(Local Broadcast

  • 背景

           Android中的广播可以跨APP(即应用程序)直接通信(exported对于有intent-filter情况下默认值为true)

           这样就会引发出以下问题:

                (1)其他APP针对性发出与当前APP intent-filter相匹配的广播,由此导致当前APP不断接收广播并处理;

                (2)其他APP注册与当前APP一致的intent-filter用于接收广播,获取广播具体信息,即会出现安全性 & 效率性问题;

  • 解决方案

          使用本地广播 || APP应用内的广播(Local Broadcast)

                (1)本地广播可理解为一种局部广播,广播的发送者与接收者都属于同一个APP;

                (2)相比于全局广播(标准广播),APP应用内广播优势体现在:安全性高 & 效率高

  • 具体使用(方法一):

             使用方式上与全局广播几乎相同,只是注册/注销广播接收者和发送广播时将参数的context变成了LocalBroadcastManager                 的单一实例。

                (1)首先还是定义广播接收者

public class LocalReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "我是一个本地广播接收者!", Toast.LENGTH_SHORT).show();
    }
}

                (2)manifest清单文件保持默认,与全局广播有所区别,因为本地广播只能通过动态注册,exported属性为true,也没                               事,本人亲自试验过,并无影响本地广播的特点,其他应用程序依然无法接收到。

        <receiver
            android:name=".LocalReceiver"
            android:enabled="true"
            android:exported="true">    
        </receiver>

                 (3)获取LocalBroadcastManager的实例,sendBroadcast的时候要使用LocalBroadcastManager类下的方法,注册注销                            广播的时候,依然是和之前动态注册的步骤一样,只不过在注册的时候,要使用LocalBroadcastManager类下的方法

public class MainActivity extends AppCompatActivity {
    private IntentFilter mIntentFilter;
    private LocalReceiver mLocalReceiver;
    private LocalBroadcastManager mLocalBroadcastManager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mLocalBroadcastManager = LocalBroadcastManager.getInstance(this);
        Button button = findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent("LOCAL_BROADCAST");
                mLocalBroadcastManager.sendBroadcast(intent);
            }
        });
        mIntentFilter = new IntentFilter();
        mIntentFilter.addAction("LOCAL_BROADCAST");
        mLocalReceiver = new LocalReceiver();
        mLocalBroadcastManager.registerReceiver(mLocalReceiver, mIntentFilter);
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        unregisterReceiver(mLocalReceiver);
    }
}

注:

       (1)对于LocalBroadcastManager方式发送的应用内广播,只能通过LocalBroadcastManager动态注册,不能静态注册,原因其实也很容易就能理解,因为静态注册主要就是为了让程序在未启动的情况下也能收到广播,而发送本地广播时,我们的程序肯定是已经启动了,因此也完全不需要使用静态注册的功能;

       (2)这里在onCreate和onDestroy,是参照《第一行代码》的案例,大家最好还是注册的时候,按照在onResume&onPause中进行注册与注销,虽然也有人指出并无关系,原因参考我上面静态与动态注册的说法,里面有给出说明 ;

  • 具体使用(方法二):

                (1)在manifest清单文件中将注册的广播中exported属性设置为false,使非本APP内部发出的广播不被其他应用的接收者                           接收;

                (2)在广播发送和接收时,增设相应权限permission,用于权限验证;

                (3)发送广播时指定该广播接收者所在的包名,此广播将只会发送到这个包中的APP内与之相匹配的有效广播接收者中;

                         通过intent.setPackage(packageName)指定包名

5. 粘性广播(Sticky Broadcast

由于在Android5.0 & API21 中已经失效,所以不建议使用,在这里也不作过多的总结。

六、BoradcastReceiver(广播)的特别注意

对于不同注册方式的广播接收器回调OnReceive(Context context,Intent intent)中的context返回值是不一样的:

  • 对于静态注册(全局+应用内广播),回调OnReceive(context,intent)中的context返回值是:ReceiverRestrictedContext
  • 对于动态注册(全局广播),回调OnReceive(context,intent)中的context返回值是:

       ActivityContext

  • 对于动态注册(本地广播LocalBroadcastManager方式),回调OnReceive(context,intent)中的context返回值是:ApplicationContext
  • 对于动态注册(本地广播非LocalBroadcastManager方式),回调OnReceive(context,intent)中的context返回值是:ActivityContext

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值