Broadcast

作者: LazyerRookie | 来源:发表于2018-06-11 21:20 被阅读0次

    一、广播是什么

    在Android中,它是一种广泛运用在应用程序之间传输信息的机制,Android中我们发送广播内容是一个Intent,这个Intent中可以携带我们要发送的数据。

    二、广播的使用场景

    • 同一app之间不同进程之间通信
    • 不同app之间组件间的通信

    三、广播的种类

    • 标准广播:一种完全异步执行的广播,广播发出之后所有广播接收器几乎会在同一时刻收到该广播,他们之间没有任何顺序。context.sendBroadcast(Intent)方法发送的广播,不可被拦截
    • 有序广播:一种同步执行的广播,在广播发出后同一时刻只会有一个广播接收器收到该广播,当这个广播接收器执行完相应逻辑后,广播才会继续传递,有先后顺序,优先级高的广播就可以先接收到广播消息并且可以截断这个广播。context.sendOrderBroadcast(Intent)方法发送的广播,可被拦截
    • 本地广播:localBroadcastManager.sendBroadcast(Intent)只在app内传播

    四、广播接收器

    广播接收器是专门用来接收广播信息的,分为静态注册和动态注册两种:

    • 静态注册:注册完成后就一直在运行
      Enabled属性:是否启用这个广播接收器
      Exported属性:是否允许该广播接收器接收本程序以外的广播
    public class BootCompleteReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"广播接收器",Toast.LENGTH_SHORT).show();
        }
    }
    

    静态注册的广播接收器一定要在AndroidManifest.xml文件中注册后才可以使用,注册代码如下所示:

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="com.example.strive.broadcast">
        <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
        <application
            android:allowBackup="true"
            android:icon="@mipmap/ic_launcher"
            android:label="@string/app_name"
            android:roundIcon="@mipmap/ic_launcher_round"
            android:supportsRtl="true"
            android:theme="@style/AppTheme">
            <activity android:name=".MainActivity">
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
            <receiver
                android:name=".BootCompleteReceiver"
                android:enabled="true"
                android:exported="true">
               <intent-filter>
                   <action android:name="android.intent.action.BOOT_COMPLETED"/>
               </intent-filter>
            </receiver>
        </application>
    </manifest>
    
    • 动态注册广播:跟随Activity的生命周期
      优点:动态注册的广播接收器可以自由地控制注册与注销,在灵活性方面有很大优势
      缺点:必须要在程序启动之后才能接收到广播,因为注册的逻辑是写在onCreate()方法中
    public class MainActivity extends AppCompatActivity {
        private IntentFilter intentFilter;
        private NetWorkChangeRecriver netWorkChangeRecriver;
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            intentFilter = new IntentFilter();
            intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
            netWorkChangeRecriver = new NetWorkChangeRecriver();
           //注册广播接收器
            registerReceiver(netWorkChangeRecriver,intentFilter);
        }
        @Override
        protected void onDestroy() {
            super.onDestroy();
            //取消注册广播接收器
            unregisterReceiver(netWorkChangeRecriver);
        }
        class NetWorkChangeRecriver extends BroadcastReceiver{
            @Override
            public void onReceive(Context context, Intent intent) {
                //获取系统服务类,专门用于管理网络连接
                ConnectivityManager connectivityManager = (ConnectivityManager) getSystemService(context.CONNECTIVITY_SERVICE);
                NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
                if(!("".equals(networkInfo))&&networkInfo.isAvailable()){
                    Toast.makeText(context, "网络连接正常", Toast.LENGTH_SHORT).show();
                }else{
                    Toast.makeText(context, "网络连接失败", Toast.LENGTH_SHORT).show();
                }
            }
        }
    }
    

    五、广播内部实现机制

    • 自定义广播接收者BroadcastReceiver,并且重写onReceiver()方法。
    • 通过Binder机制向AMS(Activity Manager Service)进行注册。
    • 广播发送者通过Binder机制向AMS发送广播。
    • .AMS查找符合条件(IntentFilter/Permission等)的BroadcastReceiver,将广播发送到相应的BroadcastReceiver(一般情况下是Activity)的消息队列中。
    • 消息循环执行拿到此广播,回调BroadcastReceiver中的onReceiver()方法。

    六、本地广播

    为了解决广播安全性的问题,Android引入了一套本地广播机制,使用这个机制发出的广播只能够在应用程序内部进行传递,并且广播接收器也只能接受应用程序内部的广播。
    本地广播的发送和注册广播接收器都需要使用LocalBroadcastManager来对广播进行管理,并提供发送广播和注册广播接收器的方法。

    public class MainActivity extends AppCompatActivity {
       private IntentFilter intentFilter;
       private LocalBroadcastManager localBroadcastManager;
       private LocalReceiver localReceier;
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
    
            localBroadcastManager = LocalBroadcastManager.getInstance(this);
    
            intentFilter = new IntentFilter();
            intentFilter.addAction("com.example.broadcasttest.LOCAL_BROADCAST");
            localReceier = new LocalReceiver();
            localBroadcastManager.registerReceiver(localReceier,intentFilter);
        }
        @Override
        protected void onDestroy() {
            super.onDestroy();
            localBroadcastManager.unregisterReceiver(localReceier);
        }
        class LocalReceiver extends BroadcastReceiver{
    
            @Override
            public void onReceive(Context context, Intent intent) {
                Toast.makeText(context, "localbroadcast", Toast.LENGTH_SHORT).show();
            }
        }
    }
    

    本地广播的特点

    • 使用它发送的广播将只在自身app内传播,因此你不必担心泄漏隐私的数据。
    • 其他app无法对你的app发送该广播,因此你的app根本不可能收到非自身app发送的该广播,因此你不必担心有安全漏洞可以利用。
    • 发送本地广播比发送系统广播更加的高效。

    本地广播的内部实现机制

    • LocalBroadcast高效的原因:因为它内部是通过Handler实现的,它的sendBroadcast()方法含义并非和系统的sendBroadcast()一样,它的sendBroadcast()方法其实就是通过Handler发送了一个Message而已。
    • LocalBroadcast安全的原因:既然它是通过Handler实现广播发送的,那么相比系统广播通过Binder机制实现那肯定更加高效,同时使用Handler来实现,别的app无法向我们应用发送该广播,而我们app内部发送的广播也不会离开我们的app。
    • LocalBroadcast内部协作主要是靠两个Map集合:mReceivers和mActions,当然还有一个List集合mPendingBroadcasts,这个主要存储待接收的广播对象。

    相关文章

      网友评论

          本文标题:Broadcast

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