「第一行代码」五、详解广播机制

广播机制简介

Android中的每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收到自己所关心的广播内容,这些广播可能来自于系统,也可能来自于其它应用程序。

发送广播的方法就是借助Intent接收广播的方法则是广播接收器(Broadcast Receiver)

 

Android中的广播主要分为两种类型:标准广播有序广播

  • 标准广播(Normal broadcasts):一种完全异步执行的广播,在广播发出后,所有的广播接收器几乎都会在同一个时刻接收到这条广播消息,因此它们之间没有任何先后顺序可言。这种广播效率较高,但无法被截断。
  • 有序广播(Ordered broadcasts):一种同步执行的广播,在广播发出后,同一时刻只会有一个广播接收器能够收到这条广播消息,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。此时的广播接收器是有先后顺序的,优先级高的广播接收器就可以先收到广播消息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接收器就无法收到广播消息了。

 

接收系统广播

广播接收器可以对自己感兴趣的广播进行注册,这样当有相应的广播发出时,广播接收器就能够收到该广播,并在内部处理相应的逻辑。注册广播的方式一般有两种:动态注册(在代码中注册)静态注册(在AndroidManifest.xml中注册)

 

动态注册

新建一个类,让它继承自BroadcastReceiver,并重写父类的onReceive()方法就行了。当有广播到来的时候,onReceive()方法就会得到执行,具体的逻辑就可以在这个方法中处理。

public class MainActivity extends AppCompatActivity {

    private IntentFilter intentFilter;
    private NetworkChangeReceiver networkChangeReceiver;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        intentFilter = new IntentFilter();
        intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
        networkChangeReceiver = new NetworkChangeReceiver();
        registerReceiver(networkChangeReceiver, intentFilter);
    }

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

    class NetworkChangeReceiver extends BroadcastReceiver {

        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context, "network changes", Toast.LENGTH_SHORT).show();
        }
    }
}

在MainActivity中定义了一个内部类NetworkChangeReceiver,这个类继承自BroadcastReceiver,并重写了父类的onReceive()方法。这样每当网络状态发生变化时,onReceive()方法就会得到执行。

在onCreate()方法中,首先创建了一个IntentFilter的实例,并给它添加了一个值为android.net.conn.CONNECTIVITY_CHANGE的action。因为当网络状态发生变化时,系统发出一条值为android.net.conn.CONNECTIVITY_CHANGE的广播。也就是说我们的广播接收器想要监听什么广播,就在这里添加对应的action。接下来创建一个NetworkChangeReceiver的实例。然后调用registerReceiver()方法进行注册将NetworkChangeReceiver的实例和IntentFilter的实例都传进去。这样NetworkChangeReceiver就会收到所有值为android.net.conn.CONNECTIVITY_CHANGE的广播。

动态注册的广播接收器一定要取消注册,这里是在onDestroy()方法中调用unregisterReceiver()方法实现的。

 

如果程序需要进行一些对用户来说比较敏感的操作,就必须在配置文件中声明权限,否则程序会崩溃。

比如访问系统的网络状态就需要声明权限。打开AndroidManifest.xml文件。

在Android 6.0系统中引入了更加严格的运行时权限。

 

静态注册

动态注册的广播接收器可以自由的控制注册与注销,在灵活性方面有很大的优势,但必须在程序启动之后才能接受广播,因为注册逻辑是写在onCreate()中的。

这里准备让程序接收一条开机广播,当收到这条广播时就可以在onReceive()方法里执行相应的逻辑,从而实现开机启动的功能。

使用Android Studio提供的快捷方式来创建一个广播接收器,右键com.cunxie.broadcasttest包 - New - Other - Broadcast Receiver。Exported属性表示是否允许这个广播接收器接收本程序以外的广播,Enabled属性表示是否启用这个广播接收器。

public class BootCompleteReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        // TODO: This method is called when the BroadcastReceiver is receiving
        // an Intent broadcast.
        Toast.makeText(context, "Boot Complete...", Toast.LENGTH_SHORT).show();
    }
}

静态的广播接收器需要在AndroidManifest.xml文件中注册。

        
            
                
            
        

标签内出现新的标签,所有静态的广播接收器都是在这里注册的。它的用法和标签非常翔实,也是通过android:name来指定具体注册哪一个广播接收器。而enabled和exported属性则是根据刚才勾选的状态自动生成。

 

不要在onReceive()方法中添加过多的逻辑或者进行任何耗时的操作,因为在广播接收器中是不允许开启线程的,当onReceive(0方法运行了较长时间而没有结束时,程序就会报错。

 

Android 8.0对广播施加了一些限制

https://developer.android.google.cn/about/versions/oreo/background

「第一行代码」五、详解广播机制_第1张图片

其中比较大的改动为:不能在AndroidManifest.xml(“清单”)中注册隐式广播接收器了。隐式广播是一种不专门针对该应用的广播。其中,系统广播都是隐式广播。当然,需要签名权限的,以及一些豁免的隐式广播仍可以在清单中注册使用。

所以:

  • 尽量使用动态注册代替静态注册。同时,动态注册的广播器依然可以监听隐式广播,包括系统广播。
  • 静态注册依然可用。在AndroidManifest.xml中依然可以注册广播接收器,不过只能监听豁免的广播或自定义的广播。

其中,针对自定义广播的接收器,需要注意:

                Intent intent = new Intent("com.cunxie.broadcasttest.MY_BROADCAST");
                intent.setComponent(new ComponentName("com.cunxie.broadcasttest",
                        "com.cunxie.broadcasttest.MyBroadcastReceiver"));
                sendBroadcast(intent);

一定要调用Intent对象的setComponent()方法,且必须在sendBroadcast(intent)之前

new ComponentName()

作用:

用于指明接收器的绝对路径

参数:

1. 广播接收器应用的包名。此处为MyBroadcastReceiver的包名

2. 广播接收器所在的完整路径。此处为MyBroadcastReceiver的完整路径

 

为了简单起见,Android 8.0+推荐使用动态注册

 

发送自定义广播

广播主要分为标准广播(Normal broadcasts)有序广播(Ordered broadcasts)

 

发送标准广播

在发送广播之前,先定义一个广播接收器来接收此广播:

public class MyBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_SHORT).show();
    }
}

在AndroidManifest中对这个广播接收器进行修改:

        
            
                
            
        

在布局文件中定义一个按钮,用于作为发送广播的触发点。然后修改MainActivity中的代码:

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button button = (Button) findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent("com.cunxie.broadcasttest.MY_BROADCAST");
                //Android 8.0+
                //new ComponentName(接收器应用的包名,接收器的绝对路径)
                intent.setComponent(new ComponentName("com.cunxie.broadcasttest",
                        "com.cunxie.broadcasttest.MyBroadcastReceiver"));
                sendBroadcast(intent);
            }
        });
    }

 

发送有序广播

发送有序广播只需要改动一行代码,即将sendBroadcast()方法改成sendOrderedBroadcast()方法

            @Override
            public void onClick(View v) {
                Intent intent = new Intent("com.cunxie.broadcasttest.MY_BROADCAST");
                //发送有序广播
                sendOrderedBroadcast(intent, null);
            }

sendOrderedBroadcast()

作用:

发送有序广播

参数:

1. Intent对象

2. 与权限有关的字符串。这里传入null即可

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

        //动态注册
        intentFilter = new IntentFilter();
        //设置优先级为100
        intentFilter.setPriority(100);
        intentFilter.addAction("com.cunxie.broadcasttest.MY_BROADCAST");
        myBroadcastReceiver = new MyBroadcastReceiver();
        registerReceiver(myBroadcastReceiver, intentFilter);

        Button button = (Button) findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent("com.cunxie.broadcasttest.MY_BROADCAST");
                //发送有序广播
                sendOrderedBroadcast(intent, null);
            }
        });
    }

通过intentFilter.setPriority()方法给广播接收器设置了优先级。

public class MyBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_SHORT).show();
        //截断广播
        abortBroadcast();
    }
}

调用了abortBroadcast()方法,表示将这条广播截断。

 

使用本地广播

以上发送和接收的广播都属于系统全局广播,即发出的广播可以被其他任何应用程序接收到,并且我们也可以接收来自于其他任何应用程序的广播。这样容易引起安全问题。

为了能够简单的解决广播的安全性问题,Android引入了本地广播机制。使用这个机制发出的广播只能够在应用程序的内部进行传递,并且广播接收器也只能接受来自本应用程序发出的广播。

主要使用了一个LocalBroadcastManager来对广播进行管理,并提供了发送广播注册广播接收器的方法。

public class MainActivity extends AppCompatActivity {

    private IntentFilter intentFilter;

    private LocalBroadcastManager localBroadcastManager;
    private LocalReceiver localReceiver;

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

        //获取实例
        localBroadcastManager = LocalBroadcastManager.getInstance(this);

        Button button = (Button) findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent("com.cunxie.broadcasttest.LOCAL_BROADCAST");
                //发送本地广播
                localBroadcastManager.sendBroadcast(intent);
            }
        });

        intentFilter = new IntentFilter();
        intentFilter.addAction("com.cunxie.broadcasttest.LOCAL_BROADCAST");
        localReceiver = new LocalReceiver();
        //注册本地广播监听器
        localBroadcastManager.registerReceiver(localReceiver, intentFilter);
    }

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

    class LocalReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context, "received local broadcast", Toast.LENGTH_SHORT).show();
        }
    }
}

首先是通过LocalBroadcastManager的getInstance()方法得到它的一个实例,然后在注册广播接收器的时候调用LocalBroadcastManager的registerReceiver()方法,在发送广播的时候调用的是LocalBroadcastManager的sendBroadcast()方法。

 

本地广播无法通过静态注册的方式来接收的。

本地广播的优势:

  • 可以明确的知道正在发送的广播不会离开我们的程序,因此不必担心机密数据泄露。
  • 其他的程序无法将广播发送到我们程序内部,因此不需要担心会有安全漏洞的隐患。
  • 发送本地广播比发送系统全局广播将会更加高效。

 

 

你可能感兴趣的:(Android)