一、广播机制
1、概念
广播机制是用来互相传递信息的一种组件
2、组成部分
发送者(sendBroadcast())
通过设置Action,标识该广播可被哪些接收器收到。
可通过putExtra,传递额外的bndle信息
接收者(BroadcastReceiver.class )
new 或者 Broadcasteceiver 类 重写 onReceive方法
在该方法中处理收到的广播消息
创建完广播接收器后,最后还需要将其注册,这样才能接收广播
分为 (静态注册)和(动态注册)
3、代码部分
(1)标识字符

private  static final String KEY_RESULT = "key_send1";
    private  String ACTION_UPLOAD_RESULT = "my_broadcast1";

(2)发送方法

public void sendBroadcast(View view) {
        Intent intent = new Intent();
        intent.setAction(ACTION_UPLOAD_RESULT);
        intent.putExtra(KEY_RESULT,"这是发送的广播1");
        sendBroadcast(intent);
    }

(3)接收类

BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver(){
        @Override
        public void onReceive(Context context, Intent intent) {
            if(intent!=null){
                String extra = intent.getStringExtra(KEY_RESULT);
                Toast.makeText(context, "收到广播:"+extra,Toast.LENGTH_SHORT).show();
            }
        }
    };

(3.1)动态注册

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

        //动态注册
        IntentFilter intentFilter = new IntentFilter();
        intentFilter.addAction(ACTION_UPLOAD_RESULT);
        registerReceiver(mBroadcastReceiver,intentFilter);
    }

(3.2)解除注册

@Override
    protected void onDestroy() {
        super.onDestroy();
        //解除注册
        unregisterReceiver(mBroadcastReceiver);
    }

以上是基于组件内的广播,组件与组件之间只需要标识符对上即可

二、静态广播和动态广播
1.概念
静态广播,是在Manifest文件中注册的广播。
常驻内存中,可在App未启动时就监听广播,如监听短信、系统时间等系统广播事件。

动态广播,是在代码中注册以及解除注册的广播
App启动后注册,然后才能监听。解除注册后广播也随着结束。

注意:同时注册时,动态广播优先于静态广播

2.静态广播的注册与使用
(1)首先,创建广播接收器类:MyBroadcastReceiver.java
重写onReceive方法

package com.example.myapplication;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.widget.Toast;

public class MyBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "收到静态广播", Toast.LENGTH_SHORT).show();
    }
}

(2)在Manifest文件中注册广播接收器

<receiver android:name=".MyBroadcastReceiver"
            android:exported="true">
            <intent-filter>
                <action android:name="my_broadcast_receive"/>
            </intent-filter>
        </receiver>

(3)使用Intent发送广播

public void sendStaticBroadcast(View view) {
        Intent intent = new Intent();
        intent.setPackage(getPackageName());
       // intent.setAction("my_broadcast_receive");
        intent.setFlags(Intent.FLAG_RECEIVER_FOREGROUND);
        sendBroadcast(intent);
    }

(4)广播接收器的onReceive方法便收到了广播信息

注意:Android 8.0及以上禁止了后台执行,因此无法收到静态注册的隐式广播。
解决方法:
1.发送广播时改为显式广播:intent.setPackage(getPackageName())
2.添加可后台执行的flag:intent.addFlags(0x1000000)
public static final int FLAG_RECEIVER_INCLUDE_BACKGROUND=0x01000000;

开机时获得静态广播

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
........................................................................................
........................................................................................
        <receiver android:name=".MyBroadcastReceiver"
            android:exported="true">
            <intent-filter>
                <action android:name="my_broadcast_receive"/>
                <action android:name="android.intent.action.BOOT_COMPLETED"/> 
            </intent-filter>
        </receiver>

三、无序广播与有序广播的区别
无序广播,又叫普通广播。所以的接收器接收的广播没有先后顺序,几乎同时收到消息。
有序广播,发送的广播会按接收器的优先级顺序被接收。同一时刻只会有一个接收器收到广播,且可以对广播截断,修改。

1.有序广播的使用
(1)动态
在注册时,进行配置

IntentFilter intentFilter = new IntentFilter();
        intentFilter.addAction(ACTION_UPLOAD_RESULT);
        intentFilter.setPriority(100);//优先级(-1000,1000)
        registerReceiver(mBroadcastReceiver,intentFilter);

(2)静态
在注册时

<receiver android:name=".MyBroadcastReceiver"
            android:exported="true">
            <intent-filter android:priority="100">
                <action android:name="my_broadcast_receive"/>
                <action android:name="android.intent.action.BOOT_COMPLETED"/>
            </intent-filter>
        </receiver>

(3)发送的时候需要用sendOrderedBroadcast()

public void sendStaticBroadcast(View view) {
        Intent intent = new Intent();
        intent.setPackage(getPackageName());
       // intent.setAction("my_broadcast_receive");
        intent.setFlags(Intent.FLAG_RECEIVER_FOREGROUND);

        sendOrderedBroadcast(intent,null);
    }

2.截断
通常使用abortBroadcast();

@Override
    public void onReceive(Context context, Intent intent) {
        Log.d(TAG, "onReceive: ");
        Toast.makeText(context, "收到静态广播", Toast.LENGTH_SHORT).show();
        abortBroadcast();
    }

3.修改有序广播
(1)通过setResultExtras(bundle)方法,向下游广播接收器传递额外键值对儿信息。
下游广播接收器通过getResultExtras方法接收信息

//发送广播
    public void sendStaticBroadcast(View view) {
        Intent intent = new Intent();
        intent.setPackage(getPackageName());
       // intent.setAction("my_broadcast_receive");
        intent.setFlags(Intent.FLAG_RECEIVER_FOREGROUND);

        Bundle bundle = new Bundle(); //创建一个Bundle 方便后续修改
        bundle.putString("name","这是有序广播");
        intent.putExtras(bundle);
        sendOrderedBroadcast(intent,null);
    }

第一个接收到广播时,对内容进行修改

public class MyBroadcastReceiver3 extends BroadcastReceiver {
    private static final String TAG = "MyBroadcastReceiver3";
    @Override
    public void onReceive(Context context, Intent intent) {
        if(intent!=null){
            Bundle dataBundle = intent.getExtras();//获取Bundle
            Log.d(TAG, "onReceive: ");
            Toast.makeText(context, "收到静态广播", Toast.LENGTH_SHORT).show();

            dataBundle.putString("name","广播3"); //对数值进行修改
            setResultExtras(dataBundle);
        }
    }
}

接收时获取被修改的信息

public class MyBroadcastReceiver2 extends BroadcastReceiver {
    private static final String TAG = "MyBroadcastReceiver2";
    @Override
    public void onReceive(Context context, Intent intent) {


        Bundle resultExtras = getResultExtras(true);
        String name = resultExtras.getString("name");//接收数据
        Toast.makeText(context, "收到静态广播", Toast.LENGTH_SHORT).show();
        Log.d(TAG, "onReceive: " + name);
    }
}

(2)通过setResultData(str)方法,向下游广播接收器传递额外字符串信息
下游广播接收器通过getResultData方法接收信息

@Override
    public void onReceive(Context context, Intent intent) {
        if(intent!=null){
            Bundle dataBundle = intent.getExtras();
            Log.d(TAG, "onReceive: ");
            Toast.makeText(context, "收到静态广播", Toast.LENGTH_SHORT).show();
            setResultData("这是广播3啊");
            setResultExtras(dataBundle);
        }
    }
@Override
    public void onReceive(Context context, Intent intent) {
        Bundle resultExtras = getResultExtras(true);
        String resultData = getResultData();
        Toast.makeText(context, "收到静态广播", Toast.LENGTH_SHORT).show();
        Log.d(TAG, "onReceive: " + resultData);
    }

四、本地广播和全局广播
本地广播,仅在本地App内部传播,其他App收不到,保证了数据的安全性
全局广播,可以在整个手机所以App之间传播,会有安全性问题。普通广播默认就是全局广播。
1.本地广播的创建 (本地广播只能动态创建)
(1)创建类

package com.example.myapplication;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

public class MyLocalBroadcastReceiver extends BroadcastReceiver {
    private static final String TAG = "MyLocalBroadcastReceive";
    @Override
    public void onReceive(Context context, Intent intent) {
        Log.d(TAG, "onReceive: 收到广播");   
    }
}

(2)引入引用

implementation("androidx.localbroadcastmanager:localbroadcastmanager:1.1.0")

(3)编写发送代码和动态注册

package com.example.myapplication;

import androidx.appcompat.app.AppCompatActivity;
import androidx.localbroadcastmanager.content.LocalBroadcastManager;

import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.view.View;

public class MainActivity extends AppCompatActivity {

    private MyLocalBroadcastReceiver myLocalBroadcastReceiver = new MyLocalBroadcastReceiver();
    private LocalBroadcastManager mLoacalBroadcastManager;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        //实例化
        mLoacalBroadcastManager = LocalBroadcastManager.getInstance(this);
        //注册广播接收器
        IntentFilter intentFilter = new IntentFilter();
        intentFilter.addAction("local_broad_cast");
        mLoacalBroadcastManager.registerReceiver(myLocalBroadcastReceiver,intentFilter);
    }

    public void sendStaticBroadcast(View view) {
        Intent intent = new Intent();
        intent.setAction("local_broad_cast");
        mLoacalBroadcastManager.sendBroadcast(intent);
    }

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

五、常用的系统广播

intent.action.AIRPLANE_MODE;
//关闭或打开飞行模式时的广播

Intent.ACTION_BATTERY_CHANGED;
//充电状态,或者电池的电量发生变化
//电池的充电状态、电荷级别改变,不能通过组建声明接收这个广播,只有通过Context.registerReceiver()注册

Intent.ACTION_BATTERY_LOW;
//表示电池电量低

Intent.ACTION_BATTERY_OKAY;
//表示电池电量充足,即从电池电量低变化到饱满时会发出广播

Intent.ACTION_BOOT_COMPLETED;
//在系统启动完成后,这个动作被广播一次(只有一次)。

Intent.ACTION_CAMERA_BUTTON;
//按下照相时的拍照按键(硬件按键)时发出的广播

Intent.ACTION_CLOSE_SYSTEM_DIALOGS;
//当屏幕超时进行锁屏时,当用户按下电源按钮,长按或短按(不管有没跳出话框),进行锁屏时,android系统都会广播此Action消息

Intent.ACTION_CONFIGURATION_CHANGED;
//设备当前设置被改变时发出的广播(包括的改变:界面语言,设备方向,等,请参考Configuration.java)

Intent.ACTION_DATE_CHANGED;
//设备日期发生改变时会发出此广播

Intent.ACTION_DEVICE_STORAGE_LOW;
//设备内存不足时发出的广播,此广播只能由系统使用,其它APP不可用?

Intent.ACTION_DEVICE_STORAGE_OK;
//设备内存从不足到充足时发出的广播,此广播只能由系统使用,其它APP不可用?

Intent.ACTION_DOCK_EVENT;
//
//发出此广播的地方frameworks\base\services\java\com\android\server\DockObserver.java

Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE;
移动APP完成之后,发出的广播(移动是指:APP2SD)

Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE;
//正在移动APP时,发出的广播(移动是指:APP2SD)

Intent.ACTION_GTALK_SERVICE_CONNECTED;
//Gtalk已建立连接时发出的广播

Intent.ACTION_GTALK_SERVICE_DISCONNECTED;
//Gtalk已断开连接时发出的广播

Intent.ACTION_HEADSET_PLUG;
//在耳机口上插入耳机时发出的广播

Intent.ACTION_INPUT_METHOD_CHANGED;
//改变输入法时发出的广播

Intent.ACTION_LOCALE_CHANGED;
//设备当前区域设置已更改时发出的广播

Intent.ACTION_MANAGE_PACKAGE_STORAGE;
//

Intent.ACTION_MEDIA_BAD_REMOVAL;
//未正确移除SD卡(正确移除SD卡的方法:设置--SD卡和设备内存--卸载SD卡),但已把SD卡取出来时发出的广播
//广播:扩展介质(扩展卡)已经从 SD 卡插槽拔出,但是挂载点 (mount point) 还没解除 (unmount)

Intent.ACTION_MEDIA_BUTTON;
//按下"Media Button" 按键时发出的广播,假如有"Media Button" 按键的话(硬件按键)

Intent.ACTION_MEDIA_CHECKING;
//插入外部储存装置,比如SD卡时,系统会检验SD卡,此时发出的广播?
Intent.ACTION_MEDIA_EJECT;
//已拔掉外部大容量储存设备发出的广播(比如SD卡,或移动硬盘),不管有没有正确卸载都会发出此广播?
//广播:用户想要移除扩展介质(拔掉扩展卡)。
Intent.ACTION_MEDIA_MOUNTED;
//插入SD卡并且已正确安装(识别)时发出的广播
//广播:扩展介质被插入,而且已经被挂载。
Intent.ACTION_MEDIA_NOFS;
//
Intent.ACTION_MEDIA_REMOVED;
//外部储存设备已被移除,不管有没正确卸载,都会发出此广播?
// 广播:扩展介质被移除。
Intent.ACTION_MEDIA_SCANNER_FINISHED;
//广播:已经扫描完介质的一个目录
Intent.ACTION_MEDIA_SCANNER_SCAN_FILE;
//
Intent.ACTION_MEDIA_SCANNER_STARTED;
//广播:开始扫描介质的一个目录

Intent.ACTION_MEDIA_SHARED;
// 广播:扩展介质的挂载被解除 (unmount),因为它已经作为 USB 大容量存储被共享。
 Intent.ACTION_MEDIA_UNMOUNTABLE;
//
Intent.ACTION_MEDIA_UNMOUNTED
// 广播:扩展介质存在,但是还没有被挂载 (mount)。
Intent.ACTION_NEW_OUTGOING_CALL;

Intent.ACTION_PACKAGE_ADDED;
//成功的安装APK之后
//广播:设备上新安装了一个应用程序包。
//一个新应用包已经安装在设备上,数据包括包名(最新安装的包程序不能接收到这个广播)
 Intent.ACTION_PACKAGE_CHANGED;
//一个已存在的应用程序包已经改变,包括包名
Intent.ACTION_PACKAGE_DATA_CLEARED;
//清除一个应用程序的数据时发出的广播(在设置--应用管理--选中某个应用,之后点清除数据时?)
//用户已经清除一个包的数据,包括包名(清除包程序不能接收到这个广播)

Intent.ACTION_PACKAGE_INSTALL;
//触发一个下载并且完成安装时发出的广播,比如在电子市场里下载应用?
//
Intent.ACTION_PACKAGE_REMOVED;
//成功的删除某个APK之后发出的广播
//一个已存在的应用程序包已经从设备上移除,包括包名(正在被安装的包程序不能接收到这个广播)

Intent.ACTION_PACKAGE_REPLACED;
//替换一个现有的安装包时发出的广播(不管现在安装的APP比之前的新还是旧,都会发出此广播?)
Intent.ACTION_PACKAGE_RESTARTED;
//用户重新开始一个包,包的所有进程将被杀死,所有与其联系的运行时间状态应该被移除,包括包名(重新开始包程序不能接收到这个广播)
Intent.ACTION_POWER_CONNECTED;
//插上外部电源时发出的广播
Intent.ACTION_POWER_DISCONNECTED;
//已断开外部电源连接时发出的广播
Intent.ACTION_PROVIDER_CHANGED;
//

Intent.ACTION_REBOOT;
//重启设备时的广播

Intent.ACTION_SCREEN_OFF;
//屏幕被关闭之后的广播

Intent.ACTION_SCREEN_ON;
//屏幕被打开之后的广播

Intent.ACTION_SHUTDOWN;
//关闭系统时发出的广播

Intent.ACTION_TIMEZONE_CHANGED;
//时区发生改变时发出的广播

Intent.ACTION_TIME_CHANGED;
//时间被设置时发出的广播

Intent.ACTION_TIME_TICK;
//广播:当前时间已经变化(正常的时间流逝)。
//当前时间改变,每分钟都发送,不能通过组件声明来接收,只有通过Context.registerReceiver()方法来注册

Intent.ACTION_UID_REMOVED;
//一个用户ID已经从系统中移除发出的广播
//

Intent.ACTION_UMS_CONNECTED;
//设备已进入USB大容量储存状态时发出的广播?

Intent.ACTION_UMS_DISCONNECTED;
//设备已从USB大容量储存状态转为正常状态时发出的广播?

Intent.ACTION_USER_PRESENT;
//

Intent.ACTION_WALLPAPER_CHANGED;
//设备墙纸已改变时发出的广播