Android广播

作者: NickelFox | 来源:发表于2017-05-06 14:35 被阅读73次

    一. 广播

    1. 广播是什么?

    广播(Broadcast)机制用于进程/线程间通信,因此在我们应用程序内发出的广播,其他的应用程序应该也是可以收到的。

    2. 广播用到的场景

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

    • 同一app内部的不同组件之间的消息通信(单个进程);

    • 同一app具有多个进程的不同组件之间的消息通信;

    • 不同app之间的组件之间消息通信;

    • Android系统在特定情况下与App之间的消息通信

    3. 广播类型

    • 标准广播:
        异步执行的广播,广播发出后所有的接收器都会接收到,没有先后顺序,无法被截断。
    • 有序广播:
        同步执行的广播,同一时刻只能有一个接收器接收到,当该接收器完成之后才能继续传播,有先后顺序,可以被接收器截断。

    二. 广播接收器(BroadcastReceiver)

    1. 什么是广播接收器?

    广播接收器是Andoird中用来接收广播(系统发送的或者其他程序发送的)的组件,是Android的四大组件之一

    2. 广播接收器类型

    • 静态广播接收器
        在AndroidManifest文件中进行注册的广播接收器,常驻型广播,也就是说当应用程序关闭后,如果有信息广播来,程序也会被系统调用自动运行。
    • 动态广播接收器
        通过Context的registerBroadcastReceiver方法进行注册的广播接收器。动态注册广播不是常驻型广播,也就是说广播跟随程序的生命周期。需要手动注销(调用unregisterBroadcastReceiver方法)。

    3. 广播接收器的用法(以下代码均为接收系统广播)

    PS:有时候接收系统广播的时候可能会需要权限,此时要记得声明权限

    • 定义一个广播接收器
    public class BootCompleteReceiver extends BroadcastReceiver {
    
        /*收到广播后会调用onReceive方法*/
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"Boot complete",Toast.LENGTH_LONG).show();
            throw new UnsupportedOperationException("Not yet implemented");
        }
    }
    
    • 声明该接收器能接收的广播(通过IntentFilter实现)
      • 静态声明(在manifest的receiver标签中配置intent-filter标签,再添加action标签)
    <receiver>
                <intent-filter>
                    <action android:name="android.intent.action.BOOT_COMPLETED"/>
                </intent-filter>
    </receiver>
    
    • 动态声明(在代码中实例化IntentFilter并为其添加action)
    IntentFilter intentFilter = new IntentFilter("android.net.conn.CONNECTIVITY_CHANGE");//实例化IntentFilter并添加action
    
    • 注册广播接收器
      • 静态注册
    <receiver
                android:name=".BootCompleteReceiver"
                android:enabled="true"
                android:exported="true">
                <intent-filter>
                    <action android:name="android.intent.action.BOOT_COMPLETED"/>
                </intent-filter>
    </receiver>
    
    • 动态注册
    mNetWorkChangeReceiver = new NetWorkChangeReceiver();//实例化BroadcastReceiver
    registerReceiver(mNetWorkChangeReceiver,intentFilter);//注册BroadcastReceiver
    
    • 动态注册一定要记得注销
    @Override
        protected void onDestroy() {
            super.onDestroy();
            unregisterReceiver(mNetWorkChangeReceiver);
        }
    

    三. 发送广播(全局广播,所有应用程序都可以接收)

    1. 简介:

    Android不仅允许系统发送广播,也允许应用程序发送广播。应用程序可以发送标准广播和有序广播。

    2. 发送方法:

    • 发送普通广播
      • 定义广播的内容(通过Intent的action)
    Intent intent = new Intent("com.foxnickel.broadcasttest.MY_BROADCAST");//定义内容为com.foxnickel.broadcasttest.MY_BROADCAST的广播
    
    • 调用sendBroadcast()方法进行发送
    sendBroadcast(intent);//发送广播
    
    • 发送有序广播
      • 定义广播的内容(通过Intent的action)
    Intent intent = new Intent("com.foxnickel.broadcasttest.MY_BROADCAST");//定义内容为com.foxnickel.broadcasttest.MY_BROADCAST的广播
    
    • 调用sendOrderedBroadcast()方法进行发送
    sendOrderedBroadcast(intent,null);//发送有序广播
    

    3. 注意

    • 广播是靠Intent来传播的,因此在发送的过程中可以通过Intent携带一些数据,然后再接收器的onReceice方法里面调用intent的get方法即可得到
    @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"received in MyBroadcastReceiver",Toast.LENGTH_SHORT).show();
            Toast.makeText(context,intent.getStringExtra("data"),Toast.LENGTH_SHORT).show();//获取intent里的数据
        }
    
    • 有序广播的接收可以设置优先级(-1000~1000)
            <receiver
                android:name=".MyBroadcastReceiver"
                android:enabled="true"
                android:exported="true">
                <intent-filter android:priority="100">//优先级低,后收到
                    <action android:name="com.foxnickel.broadcasttest.MY_BROADCAST"/>
                </intent-filter>
            </receiver>
            <receiver
                android:name=".AnotherReceiver"
                android:enabled="true"
                android:exported="true">
                <intent-filter android:priority="200">//优先级高,先收到
                    <action android:name="com.foxnickel.broadcasttest.MY_BROADCAST"/>
                </intent-filter>
            </receiver>
    

    四. 本地广播(在应用内部传递的广播)

    1. 本地广播的优势

    • 所有数据交换都在本程序内,不必担心数据泄露
    • 其他程序无法将广播发送到本程序内部
    • 更高效

    2. 本地广播要注意的地方

    • 只能动态注册,不能静态注册

    3. 本地广播的用法

    • 实例化LocalBroadcastManager
    mLocalBroadcastManager = LocalBroadcastManager.getInstance(this);
    
    • 发送广播(调用LocalBroadcastManager的sendBroadcast方法)
    Intent intent = new Intent("com.foxnickel.broadcasttest.LOCAL_BROADCAST");
    mLocalBroadcastManager.sendBroadcast(intent);//发送本地广播
    
    • 接收广播
      • 新建广播接收器(与普通的广播接收器一样)
    class LocalReceiver extends BroadcastReceiver{
    
            @Override
            public void onReceive(Context context, Intent intent) {
                Toast.makeText(context,"received local broadcast",Toast.LENGTH_SHORT).show();
            }
        }
    
    • 注册接收器,接收广播(调用LocalBroadcastManager的registerBroadcastReceiver方法)
            IntentFilter intentFilter = new IntentFilter("com.foxnickel.broadcasttest.LOCAL_BROADCAST");
            LocalReceiver localReceiver = new LocalReceiver();
            mLocalBroadcastManager.registerReceiver(localReceiver,intentFilter);
    
    • 记得销毁接收器
    @Override
        protected void onDestroy() {
            super.onDestroy();
            mLocalBroadcastManager.unregisterReceiver(localReceiver);
        }  
    

    五. 注意事项

    • 动态注册的广播接收器好像无法接收到其他应用程序发来的广播(不知道是不是自己写错了)
    • 静态注册的广播接收器即使app已经退出,主要有相应的广播发出,依然可以接收到,但此种描述自Android 3.1开始有可能不再成立(参考Android总结篇系列:Android广播机制
      • 对于系统广播,由于是系统内部直接发出,无法更改此intent flag值,因此,3.1开始对于静态注册的接收系统广播的BroadcastReceiver,如果App进程已经退出,将不能接收到广播。
      • 但是对于自定义的广播,可以通过复写此flag为FLAG_INCLUDE_STOPPED_PACKAGES,使得静态注册的BroadcastReceiver,即使所在App进程已经退出,也能能接收到广播,并会启动应用进程,但此时的BroadcastReceiver是重新新建的。

    相关文章

      网友评论

        本文标题:Android广播

        本文链接:https://www.haomeiwen.com/subject/lldzattx.html