笔记:Android Brodcast广播

广播分为两种:普通广播、有序广播
普通广播:是一种完全异步执行的广播。在广播发出之后所有的广播接收器几乎同时接受到这条消息,无法被截断
有序广播:是一种同步执行的广播,在广播发出后同时只有一个广播接收器接收到这条消息,当这个广播接收器中的逻辑执行完成后,广播才会继续传递,接受消息有先后顺序,优先级高的接收器先接收广播,并且前面的接受器可以截断正在传送的广播

接收系统广播

注册广播接收器:动态注册、静态注册

	public class NetworkChangeReceiver extends BroadcastReceiver{
        @Override
        public void onReceive(Context context, Intent intent) {
            String action = intent.getAction();
            if (action.equals("android.net.conn.CONNECTIVITY_CHANGE")){
                NetworkIsAvailAble(context);
            }else if (action.equals("com.zwt.myBrodcasts")){
                Toast.makeText(BrodcastsActivity.this,"收到自定义广播", Toast.LENGTH_LONG).show();
            }
        }

		@SuppressLint("WrongConstant")
	    private void NetworkIsAvailAble(Context context){
	        ConnectivityManager connectivityManager = (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);
	        // Android6.0 以上
	        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
	            Network notwork = connectivityManager.getActiveNetwork();
	            NetworkCapabilities networkCapabilities = connectivityManager.getNetworkCapabilities(notwork);
	            if (networkCapabilities != null &&
	                    (
	                        networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI)
	                        || networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR)
	                        || networkCapabilities.hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET)
	                    )){
	                Toast.makeText(context,"23 network is available", Toast.LENGTH_SHORT).show();
	            }else {
	                Toast.makeText(context,"23 network is unavailable", Toast.LENGTH_SHORT).show();
	            }
	        }else {
	            // Android6.0 以上弃用了该方法
	            NetworkInfo notworkInfo = connectivityManager.getActiveNetworkInfo();
	            if (notworkInfo != null && notworkInfo.isAvailable()){
	                // 网络可用
	                Toast.makeText(context,"network is available", Toast.LENGTH_SHORT).show();
	            }else {
	                // 网络不可用
	                Toast.makeText(context,"network is unavailable", Toast.LENGTH_SHORT).show();
	            }
	        }
	    }
    }

定义了一个类 NetworkChangeReceiver 继承 BroadcastReceiver 并重写父类方法 onReceive(),onReceive() 方法有两个参数,一个是上下文 context,一个是 Intent 通过 Intent.getAction() 方法获得广播的action的值,通过这个值可以判断是从哪里发出的广播
定义了一个 NetworkIsAvailAble() 方法用来判断网络是否可用
通过 getSystemService() 方法得到 ConnectivityManager 的实例,这是一个系统服务类,专门用于网络连接管理,注意 Android6.0 以上新版本 getActiveNetworkInfo() 被弃用 使用 getActiveNetwork() 来替代
添加访问网络的权限

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
动态注册
	private IntentFilter intentFilter;
    private NetworkChangeReceiver networkChangeReceiver;
    
	@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_brodcasts);

        intentFilter = new IntentFilter();
        intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
        networkChangeReceiver = new NetworkChangeReceiver();
        registerReceiver(networkChangeReceiver, intentFilter);
    }
    
    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (networkChangeReceiver!= null){
        	unregisterReceiver(networkChangeReceiver);
       	}
    }

在 onCreate() 方法中进行广播的注册,首先创建了一个 IntentFilter 的实例,并添加了一个 android.net.conn.CONNECTIVITY_CHANGE 的 action,这样应用就可以接收网络变化的广播了,然后创建了 NetworkChangeReceiver 的实例调用 registerReceiver( ) 方法进行注册
注:动态注册的广播一定要取消注册,在 onDestroy() 调用 unregisterReceiver() 方法取消注册

静态注册

动态广播需要在程序启动后才能接收到广播,静态广播可以在程序未启动下接受广播,例如接收开机广播
系统启动后会发出 android.intent.action.BOOT_COMPLETED 的广播,需要监听系统开机广播需要权限

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

静态广播在 AndroidManifest.xml 文件中注册,可以使用 Android Studio 的快捷方式创建广播接收器
New --> Other --> Broadcast Receiver
Exported 属性表示是否允许这个广播接收程序以外的广播,如果设置了 android:exported=“false”,就只能接收本应用的广播消息,Enabled 属性表示是否启用这个广播接收器
在这里插入图片描述

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.helloworld">

    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <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">
        <receiver
            android:name="com.zwt.broadcasts.NetworkChangeReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>
                <action android:name="android.intent.action.BOOT_COMPLETED"/>
            </intent-filter>
        </receiver>
</manifest>

发送自定义广播

在发送广播之前要先定义广播接收器,如上面所说(动态注册静态注册),注意动态注册要添加 intentFilter.addAction(com.zwt.myBrodcasts) 静态注册要添加 <action android:name="com.zwt.myBrodcasts"/>,这里是可以接收一条 action 值为 com.zwt.myBrodcasts 的自定义广播

发送标准广播

发送广播定义一个按钮为发送广播的出发条件

		button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Intent intent = new Intent("com.zwt.myBrodcasts");
                sendBroadcast(intent);
            }
        });

构建了一个 Intent 对象,前面说过 Intent 是可以携带数据的,所以广播也是可以传递数据,然后调用 sendBroadcast() 将广播发送出去

发送有序广播
		button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Intent intent = new Intent("com.zwt.myBrodcasts");
                sendOrderedBroadcast(intent, null);
                // FinalBroadcast是一个广播接收器,这种方法FinalBroadcast接受者就算被优先级高的广播拦截,也都会接收到广播
                //sendOrderedBroadcast(intent, null, new FinalBroadcast(), null, 0, null, null);
            }
        });

发送有序广播,即将 sendBroadcast() 改成 sendOrderedBroadcast() 方法,其接收两个参数一个是 Intent 一个是与权限相关的字符串
静态广播设置优先级:

		<receiver
            android:name="com.example.broadcasts.BootCompleteReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter android:priority="100">
                <action android:name="android.intent.action.BOOT_COMPLETED"/>
            </intent-filter>
        </receiver>

通过 intent-filter 标签中的 android:priority 属性设置
动态广播设置优先级:

protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_brodcasts);

        intentFilter = new IntentFilter();
        intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
        intentFilter.setPriority(100);
        networkChangeReceiver = new NetworkChangeReceiver();
        registerReceiver(networkChangeReceiver, intentFilter);
    }

通过 IntentFilter 对象的 setPriority() 方法设置
注意:优先级越大表示优先级越高表示越先收到广播,优先级是一个 int 类型,也就是在32位和64位电脑上最大是 2147483647,若优先级一样,那么:

  • 动态注册优先级别高于静态注册
  • 早动态注册优先级别高于后动态注册的广播
  • 早安装的程序静态注册优先级高于后安装程序静态注册广播(安装APK会解析manifest.xml,把其加入队列)
    截断广播:
		@Override
        public void onReceive(Context context, Intent intent) {
        	Toast.makeText(BrodcastsActivity.this,"收到广播", Toast.LENGTH_LONG).show();
            abortBroadcast();
        }

在广播接收器中的 onReceive() 方法中调用 abortBroadcast() 方法就表示这条广播被截断,后面的广播接收器无法收到这条广播

发送本地广播

目前发送和接受的都是全局广播,即发出的广播可以被其他任何应用接收到,并且也可以接收到任何其他程序的广播,为了预防敏感数据被其他应用程序截获,或者其他程序发送大量的垃圾广播,Android引入一套本地 广播机制
添加依赖:
在这里插入图片描述

	//private IntentFilter intentFilter;
	private LocalBroadcastManager localBroadcastManager;
    private NetworkChangeReceiver networkChangeReceiver;
    
	@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_brodcasts);

        //intentFilter = new IntentFilter();
        localBroadcastManager = LocalBroadcastManager.getInstance(this);
        intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
        networkChangeReceiver = new NetworkChangeReceiver();
        //registerReceiver(networkChangeReceiver, intentFilter);
        localBroadcastManager.registerReceiver(networkChangeReceiver, intentFilter);
    }
    
    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (networkChangeReceiver!= null){
        	//unregisterReceiver(networkChangeReceiver);
        	localBroadcastManager.unregisterReceiver(networkChangeReceiver);
       	}
    }

本地广播使用一个 LocalBroadcastManager 对广播进行管理,通过 getInstance() 方法获得实例,使用 registerReceiver() 方法进行注册,使用 unregisterReceiver() 方法进行注销

			@Override
            public void onClick(View view) {
                Intent intent = new Intent("com.zwt.myBrodcasts");
                localBroadcastManager.sendBroadcast(intent)
            }

使用 sendBroadcast() 方法发送广播,localBroadcastManager 还有一个 sendBroadcastSync() 方法是用于发送同步广播,也就是说要等广播接受者处理完intent才能往下执行,发送广播的动作还是异步多线程的,要等到所有的广播接收者执行完主线程才会向下执行
注意:本地广播不能通过静态注册的方式来接收广播

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值