【Android】广播机制

一、广播机制简介

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.发送本地广播比系统全局广播更高效

### Android 广播机制的工作原理 Android广播机制是一种基于消息传递的设计模式,允许应用之间或者同一应用内的组件间通过发送接收广播来通信。这种机制的核心在于 `BroadcastReceiver` 组件以及系统的广播分发器。 #### 1. **广播的分类** Android 中的广播分为两种主要类型:标准广播 (Normal Broadcast) 和有序广播 (Ordered Broadcast)[^2]。 - 标准广播是异步执行的,所有的广播接收者几乎会同时接收到这条广播消息。 - 有序广播则是同步执行的,按照优先级顺序依次处理广播消息,并且前一个接收者可以修改广播的内容甚至中断广播。 #### 2. **广播的注册方式** 广播可以通过静态注册或动态注册的方式实现: - 静态注册是指在 `AndroidManifest.xml` 文件中定义 `<receiver>` 节点并指定其过滤条件。 - 动态注册则是在代码运行期间调用 `Context.registerReceiver()` 方法完成注册操作。 #### 3. **广播的生命周期管理** 无论是哪种注册形式都需要特别注意资源管理和内存泄漏的风险。对于动态注册而言,在适当的时候比如 Activity 或 Service 销毁之前应该解除绑定以释放相关联的对象实例;而对于静态声明的情况,则无需手动注销因为它们依赖于进程本身的存在状态而存在。 #### 4. **具体实现步骤** 以下是创建简单本地广播的一个例子: ```java // 定义自定义动作名称常量 public static final String ACTION_MY_BROADCAST = "com.example.MY_ACTION"; // 发送广播的方法示例 Intent intent = new Intent(ACTION_MY_BROADCAST); intent.putExtra("key", value); // 可选参数设置 sendBroadcast(intent); // 接收广播的类继承 BroadcastReceiver 类 public class MyBroadcastReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (ACTION_MY_BROADCAST.equals(action)) { int receivedValue = intent.getIntExtra("key", defaultValue); Log.d(TAG, "Received broadcast with key=" + receivedValue); } } } ``` 上述代码展示了如何定义自己的广播意图、附加数据到该意图上并通过系统发出广播的过程。另外还给出了响应特定类型的广播事件所需的逻辑结构。 ### 结论 综上所述,掌握 Android 广播机制不仅有助于开发者构建更加灵活的应用程序架构,而且还能促进不同模块间的松耦合协作关系形成。然而需要注意的是,在实际项目开发过程中应当谨慎对待权限控制及性能优化等问题以免引发不必要的安全隐患或是降低用户体验质量[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值