Android广播——Broadcast使用探索

1. 广播概述

Broadcast就是应用间的全局大喇叭,即通信的一个手段。Android中每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序只会接收到自己关心的广播内容。这些广播可能来自系统,也可能来自其他应用程序。发送广播借助Intent,接收广播需要引入一个新的概念——广播接收器(Broadcast Receiver)。

BroadcastReceiver 是对发送出来的 Broadcast 进行过滤、接受和响应的组件。首先将要发送的消息和用于过滤的信息(Action,Category)装入一个 Intent 对象,然后通过调用Context.sendBroadcast()sendOrderBroadcast() 方法把 Intent 对象以广播形式发送出去。 广播发送出去后,所以已注册的 BroadcastReceiver 会检查注册时的 IntentFilter 是否与发送的 Intent 相匹配,若匹配则会调用 BroadcastReceiver 的 onReceiver()方法

所以当我们定义一个 BroadcastReceiver 的时候,都需要实现 onReceiver()方法。BroadcastReceiver 的生命周期很短,在执行 onReceiver() 方法时才有效,一旦执行完毕,该Receiver 的生命周期就结束了

1.1 广播类型

Android中广播主要分为两种类型:标准广播有序广播。
在这里插入图片描述

2. 接收广播系统

android 内置了很多系统级别的广播,可以在应用程序中通过监听这些波光来得到各种系统的状态信息。想要接收这些广播,需要使用广播接收器。注册的方法又分为两种:动态与静态

2.1 动态注册(监听网络变化)

动态注册 BroadcastReceiver 是在代码中定义并设置好一个 IntentFilter 对象,然后在需要注册的地方调用 Context.registerReceiver() 方法,调用 Context.unregisterReceiver()方法取消注册,此时就不需要在清单文件中注册 Receiver 了

创建一个广播接收器,只需要新建一个类,让它继承自BroadcastReceiver,并重写父类的onReceive()方法就可以。当有广播来,onReceive()方法就会得到执行,具体逻辑可以再这个方法中处理。

示例:(监听网络变化

  • 修改MainActivity.java代码
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) {
            ConnectivityManager connectionManager = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
            NetworkInfo networkInfo = connectionManager.getActiveNetworkInfo();
            if (networkInfo != null && networkInfo.isConnected()) {
                Toast.makeText(context, "network is Connected", Toast.LENGTH_SHORT).show();
            } else {
                Toast.makeText(context, "network is unavailable", Toast.LENGTH_SHORT).show();
            }

//            Toast.makeText(context,"network changes", Toast.LENGTH_SHORT).show();
        }
    }
}
  • 在配置文件AndroidManifest.xml文件中声明权限
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

备注

  • 在MainActivity中定义了一个内部类继承BroadcastReceiver,重写了父类的onReceive()方法。每当网络状态发生变化时,该方法就会得到执行。
  • onCreate()方法,创建了一个IntentFilter的实例,并给它添加了一个值为android.net.conn.CONNECTIVITY_CHANGE的action。当网络状态发生变化时,系统发出的正是一条值为android.net.conn.CONNETIVITY_CHANGE的广播。
  • 创建一个NetWorkChangeReceiver的实例,调用registerReceiver()方法进行注册,这样NetworkChangeReceiver就会收到所有值为
  • 动态注册的广播接收器一定要取消注册,在onDestroy()方法中调用unregisterReceiver()方法实现
  • onReceiver()方法中,首先通过getSystemService()方法得到了ConnectivityManager实例,这是一个系统服务类,专门用于管理网络连接的。
  • 调用getActiveNetworkInfo()方法可以得到NetworkInfo实例,接着调用NetworkInfoisConnected()方法,就可以判断出当前是否有网络,最后通过Toast方式对用户进行提示。

2.2 静态注册(开机启动)

程序在未启动的情况下就能接收到广播,就需要静态注册的方式。

  • 首先右击包新建一个Broadcast Receiver,勾选Exported属性与Enabled属性

    扫描二维码关注公众号,回复: 6392857 查看本文章
  • 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
        Toast.makeText(context, "Boot Complete", Toast.LENGTH_LONG).show();
    }
}
  • 在AndroidManifest.xml中注册
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
...
<receiver
            android:name=".BootCompleteReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED"/>
            </intent-filter>
        </receiver>

备注

  • android系统启动完成会发出一条值为android.intent.action.BOOT_COMPELTED的广播,在<intent-filter>中添加相应的action
  • 监听系统开机广播需要声明权限,使用<uses-permission>标签又加入一条android.permission.RECEIVE_BOOT_COMPLETED权限
  • 不能在onReceive()方法中添加过多的逻辑或者耗时操作,广播接收器中不允许开启线程,当onRecieve()方法运行了较长时间没有结束,程序就会报错。

3. 发送自定义广播

发送广播前,要先定义一个广播接收器

  • 自定义一个BroadcastReceiver,重写onReceive()方法,注册下
    • 标准广播:sendBroadcast(intent)
    • 有序广播:sendOrderedBroadcast(intent, null)
  • 在有序广播的清单文件中的Intent-filter通过:android:priority="100"设置优先级,值越大优先级越高,越先收到广播,可以调用abortBroadcast()截断广播的继续传递优先级可选值:-1000~1000之间

3.1 标准广播

  • 首先定义一个广播接收器接受此广播:
public class MyBroadcastReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        // TODO: This method is called when the BroadcastReceiver is receiving
        Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_SHORT).show();
    }
}
  • AndroidManifest.xml文件中进行注册修改
<receiver
    android:name=".MyBroadcastReceiver"
    android:enabled="true"
    android:exported="true">
    <intent-filter>
        <action android:name="com.baiheng.broadcasttest.MY_BROADCAST"/>
    </intent-filter>
</receiver>
  • activity_main.xml文件中定义一个按钮,作为发送广播触发点
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Send Broadcast"/>

</LinearLayout>
  • 修改MainActivity.java代码
public class MainActivity extends AppCompatActivity {
    //...
    @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.baiheng.broadcasttest.MY_BROADCAST");
                intent.setComponent(new ComponentName("com.baiheng.broadcasttest","com.baiheng.broadcasttest.MyBroadcastReceiver"));
                sendBroadcast(intent);
            }
        });
        //...
    }
}

备注

  • Android8.0之后,对广播做了一些限制,除了有限的例外情况,应用无法使用清单注册隐式广播。 它们仍然可以在运行时注册这些广播,并且可以使用清单注册专门针对它们的显式广播。
  • -上述使用静态注册的广播接收器接收自定义的广播,会发生接收不到自定义广播的问题,一种解决方法就是在Android7.0代码基础上在发送广播的基础上加上:
intent.setComponent(new ComponentName("com.vivo.a11085273.broadcasttest","com.vivo.a11085273.broadcasttest.MyBroadcastReceiver"));

其中ComponentName第一个参数是自定义广播的包名,第二个参数是广播接收器的类

3.2 有序广播

由于Android8.0后限制了隐式广播,现在新建一个项目BroadcastTest2,自定义一个广播接收器AnotherBroadcastReceiver,动态注册,将前述的广播也改为动态注册,使得两个应用都可以接受到广播。

  • 对于有序广播,在BroadcastTest项目中修改MainActivity中代码:
sendOrderedBroadcast(intent,null);

sendBroadcast()方法改成sendOrderedBroadcast()方法。该方法接收两个参数,第一个参数为Intent,第二个参数是一个权限相关字符。

  • 如果想要设定广播接收器优先级,并将广播截断,可以使用setPriority()方法,在动态注册处设置优先级:
intentFilter2.setPriority(100);

将MyBroadReceiver优先级设置为100,将项目BroadcastTest2中AnotherBroadcastReceiver优先级设为50,则MyBroadReceiver先收到广播

  • 如果MyBroadcastReceiver想要截断广播
public class MyBroadcastReceiver extends BroadcastReceiver {

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

4. 本地广播

Android引入了一套本地广播机制,使用这个机制广播只能够在应用程序的内部进行传递,并且广播接收器也只能接收来自应用程序发出的广播

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

核心方法(使用LocalBroadcastManager管理广播)

  • 调用LocalBraodcastManager.getInstance()获得实例
  • 调用~.registerReceiver()注册广播
  • 调用~.sendBroadcast()发送广播
  • 调用~.unregisterReceiver()取消注册
  • ~表示LocalBroadcastManager的实例

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

示例

public class MainActivity extends AppCompatActivity {

    private IntentFilter intentFilter2;
    private LocalReceiver localReceiver;

    private LocalBroadcastManager localBroadcastManager;
    
     @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.baiheng.broadcasttest.LOCALBROADCAST");
                localBroadcastManager.sendBroadcast(intent);
            }
        });
         intentFilter2 = new IntentFilter();
        intentFilter2.addAction("com.baiheng.broadcasttest.LOCALBROADCAST");
        localReceiver = new LocalReceiver();
        localBroadcastManager.registerReceiver(localReceiver, intentFilter2);//注册监听广播
    }
    
    @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();
        }
    }
}

备注

  • 本地广播无法通过静态注册来接收,比全局系统广播更高效
  • 在广播中启动Activity,需要为intent加入FLAG_ACTIVITY_NEW_TASK的标记,不然会报错,因为需要一个新的栈来存放新打开的Activity
  • 广播中弹出的AlertDialog,需要设置对话框类型为TYPE_SYSTEM_ALERT,不然无法弹出

猜你喜欢

转载自blog.csdn.net/weixin_43499030/article/details/89480513