【Android】广播机制

发布于:2025-07-27 ⋅ 阅读:(14) ⋅ 点赞:(0)

一、广播机制简介

android的应用程序可以自由的发送和接收广播,这些广播可以来自于系统和其它应用程序,Android提供了一套完整的API,允许应用自由的发送和接收广播。

广播类型

标准广播:

是一种完全异步执行的广播,所有的广播接收器几乎会在同一时间接收到广播消息,无法被截断。

有序广播:

同步执行的广播,同一时刻只有一个广播接收器能够接收到这条广播消息,广播接收器有先后顺序,优先级高的广播接收器先收到广播消息,前面的广播接收器可以截断正在传递的广播。

二、接收系统广播

2.1动态注册

在代码中对自己感兴趣的广播进行注册,动态注册只有在程序启动后才能接收到广播

以监听网络变化为例:

1.新建一个类,继承BroadcastReceiver并重写父类onReceive方法,当有广播时会执行这个方法,具体的逻辑就在这个方法中处理

2.在onCreate方法中创建IntentFilter实例,并给它添加相应广播的action

3.创建新建类的实例,调用registerReceiver方法进行注册,将新建类实例和IntentFilter实例都传进去

public class MainActivity extends AppCompatActivity {
    private IntentFilter intentFilter;
    private NetworkChangeReceiver networkChageReceiver;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        intentFilter=new IntentFilter();
        intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
        networkChageReceiver=new NetworkChangeReceiver();
        registerReceiver(networkChageReceiver,intentFilter);
    }
    protected void onDestory(){
        super.onDestroy();
        unregisterReceiver(networkChageReceiver);
    }
    class NetworkChangeReceiver extends BroadcastReceiver{
        public void onReceive(Context context, Intent intent){
            Toast.makeText(context,"network changes",Toast.LENGTH_SHORT).show();
        }
    }
}

动态注册的广播会在onDestroy中调用unregisterReceiver方法取消注册

class NetworkChangeReceiver extends BroadcastReceiver{
        public void onReceive(Context context, Intent intent){
            ConnectivityManager connectivityManager=(ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
            NetworkInfo networkInfo=connectivityManager.getActiveNetworkInfo();
            if(networkInfo!=null&&networkInfo.isAvailable()){
                Toast.makeText(context,"available",Toast.LENGTH_SHORT).show();
            }else {
                Toast.makeText(context,"unavailable",Toast.LENGTH_SHORT).show();
            }
        }
    }

使用getSystemService得到的ConnectivityManager实例是一个系统服务类,专门用于管理网络连接,之后调用getActiveNetworkInfo方法得到NetworkInfo实例,之后调用NetworkInfo的isAvailable可以判断出当前是否有网络 。

2.2静态注册

静态注册可以可以在程序未启动的情况下接收广播

1.创建一个广播接收器,修改onReceive方法。

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();
    }
}

静态广播接收器要在AndroiManifest.xml中注册

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools">

    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
    <application
        android:allowBackup="true"
        android:dataExtractionRules="@xml/data_extraction_rules"
        android:fullBackupContent="@xml/backup_rules"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.BroadcastTest"
        tools:targetApi="31">
        <receiver
            android:name=".BootCompleteReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED"/>
            </intent-filter>
        </receiver>

        <activity
            android:name=".MainActivity"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

<receiver>用法和<activity>标签相似,通过android:name来指定具体注册哪一个广播接收器

还要在 <intent-filter>标签里添加相应的action

三、发送自定义广播

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
        // an Intent broadcast.
        Toast.makeText(context,"MyBroadcastReceiver",Toast.LENGTH_SHORT).show();
    }
}

 修改广播接收器注册

  <receiver
            android:name=".MyBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter>
                <action android:name="com.example.broadcasttest.MY_BROADCAST"/>
            </intent-filter>
        </receiver>

 修改主活动代码,发送广播

 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(){
            public void onClick(View v){
                Intent intent=new Intent("com.example.broadcasttest.MY_BROADCAST");
                intent.setPackage(getPackageName());
                sendBroadcast(intent);
            }
        });
    }

Android 8.0+ 的特殊处理

对于 Android 8.0 (API 26) 及以上版本:

  • 必须为静态注册的接收器指定 package

  • intent.setPackage(getPackageName());
    

3.2发送有序广播

程序发送的广播可以被其他的应用程序接收

发送有序广播需要修改主活动

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(){
            public void onClick(View v){
                Intent intent=new Intent("com.example.broadcasttest.MY_BROADCAST");
                intent.setPackage(getPackageName());
                sendOrderedBroadcast(intent,null);
            }
        });
    }

使用 sendOrderedBroadcast,接收两个参数Intent和与权限有关的字符串

在注册时可以设置广播接收器的先后顺序

<receiver
            android:name=".MyBroadcastReceiver"
            android:enabled="true"
            android:exported="true"
            android:permission="your.custom.permission">
            <intent-filter android:priority="100">
                <action android:name="com.example.broadcasttest.MY_BROADCAST"/>
            </intent-filter>
        </receiver>

设置优先级后,优先级高的广播接收器可以选择是否允许广播继续传递

如果在onReceive方法中调用abortBroadcast(),就表示将这条广播截断

3.3本地广播

本地广播机制下发出的广播只能在应用程序的内部进行传播,广播接收器也只能接收来自本应用程序发出的广播。

使用LocalBroadcastManager对广播进行管理,提供了发送广播和注册广播接收器的方法。

public class MainActivity extends AppCompatActivity {
    private IntentFilter intentFilter;
    private   LocalReceiver receiver;
    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(){
            public void onClick(View v){
                Intent intent=new Intent("com.example.broadcasttest.MY_BROADCAST");
                localBroadcastManager.sendBroadcast(intent);
            }
        });
        intentFilter=new IntentFilter();
        intentFilter.addAction("com.example.broadcasttest.MY_BROADCAST");
        receiver=new LocalReceiver();
        localBroadcastManager.registerReceiver(receiver,intentFilter);
    }
    protected void onDestroy() {
        super.onDestroy();
        // 必须取消注册,否则内存泄漏!
        if (receiver != null) {
            unregisterReceiver(receiver);
        }
    }
    class LocalReceiver extends BroadcastReceiver{
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"local",Toast.LENGTH_SHORT).show();
        }
    }
}

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

本地广播的优势

1.不担心机密数据泄露

2.其他程序无法将广播发送到程序内部,不担心安全漏洞隐患

3.发送本地广播比系统全局广播更高效


网站公告

今日签到

点亮在社区的每一天
去签到