Android 第二十课 广播机制(大喇叭)----发送自定义广播(包括发送标准广播和发送有序广播)

19 篇文章 0 订阅
1 篇文章 0 订阅

广播分为两种类型:标准广播和有序广播

我们来看一下具体这两者的具体区别:

1、发送标准广播

我们需要先定义一个广播接收器来准备接收此广播才行,否则也是白发。

新建一个MyBroadcastReceiver,代码如下:

package com.example.broadcasttest;

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

/**
 * Created by ZHJ on 2018/3/11.
 */

public class MyBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context,"received in MyBroadcastReceiver",Toast.LENGTH_SHORT).show();
    }
}

这里当MyBroadcastReceiver收到自定义的广播时,就会弹出“received in MyBroadcastReceiver ”的提示。然后在AndroidManifest.xml中对这个广播接收器进行修改:

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

    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name = "android.permision.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">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

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

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

</manifest>

可以看到,这里让MyBroadcastReceiver接收一条值为com.example.broadcasttest.MY_BROADCAST的广播,因此待会在发送广播的时候,我们就需要发出这样的一条广播。

修改activity_main.xml中的代码,如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.example.broadcasttest.MainActivity">

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

</LinearLayout>

我们在布局中添加了一个按钮,用于作为发送广播的触发点。

然后修改MainActivity中的代码:

package com.example.broadcasttest;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {


    private IntentFilter intentfiletr;
    private NetworkChangeReceiver networkChangeReceiver;
    @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 view) {
                Intent intent = new Intent("com.example.broadcasttest.MY_BROADCAST");
                sendBroadcast(intent);
            }
        });
    }

   ....
}
可以看到,我们在按钮的点击事件里面加入了发送自定义广播的逻辑,首先构建出了一个Intent对象,并把要发送的广播的值传入,然后调用了Context的sendBroadcast()方法将广播发送出去,这样监听com.example.broadcasttest.MY_BROADCAST这条广播的广播接收器会收到消息。此时发出去的广播就是一条标准广播。

运行程序,点击按钮。

回顾如何发送一条标准广播,首先你得有个广播接收器,那我们就新建一个广播接收器MyBroadcastReceiver,然后在里面添加一个Toast,用于接收后广播用于反馈,但是我们要在AndroidManifest.xml文件中对这个广播接收器进行修改,你要接收什么样得广播。广播接收器就差不多做好了。我们开始准备发送广播,添加一个按钮,作为触发点,在按钮的点击事件中,添加发送自定义广播的逻辑。首先肯定要构建出Intent对象,把要发送的广播的值传入,然后调用Context的sendBroadcast()方法将广播发送出去。这样所有监听com.example.broadcasttest.MY_BROADCAST这条广播的广播接收器就会收到消息。这就是一条标准广播。

另外,广播是使用Intent进行传递的,因此你还可以在Intent中携带一些数据传递给广播接收器。

2、发送有序广播

广播是一种跨进程的通信方式,我们在应用程序内发出去的广播,其他应用程序也是可以接收的。废话不说,我们要验证,

新建BroadcastTest2项目。当然我们还需要在这个项目中新建一个广播接收器,用于接收上一次的自定义广播,

新建AnotherBroadcastReceiver,代码如下:

package com.example.broadcasttest2;

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

/**
 * Created by ZHJ on 2018/3/11.
 */

public class AnotherBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context,"received AnotherBroadcastReceiver",Toast.LENGTH_SHORT).show();
    }
}

我们仍然是在广播接收器的onReceive()方法中弹出了一段文本信息。然后,AndroidManifest.xml中对这个广播接收器进行修改,代码如下:

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

    <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">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        
        <receiver android:name=".AnotherBroadcastReceiver">
            <intent-filter>
                <action android:name="com.example.broadcasttest.MY_BROADCAST"/>
            </intent-filter>
        </receiver>
    </application>
    
</manifest>

可以看到,AndroidBroadcastReceiver接收的仍然是com.example.broadcasttest.MY_BROADCAST这条广播,把BroadcastTest2运行起来,点击BroadcastTest1的按钮,那么你会接收两条提示信息。

这就证明了,我们的应用程序是可以被其他的应用程序接收到的。

发送有序广播:

到现在为止,我们程序中发送的都是标准广播,接下来,我们来发送有序广播,重新回到Broadcast项目,然后修改MainActivity中的代码,如下所示:

package com.example.broadcasttest;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {


    private IntentFilter intentfiletr;
    private NetworkChangeReceiver networkChangeReceiver;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        intentfiletr = new IntentFilter();
        intentfiletr.addAction("android.net.conn.CONNECTIVITY_CHANGE");
        networkChangeReceiver = new NetworkChangeReceiver();
        registerReceiver(networkChangeReceiver,intentfiletr);


        Button button = (Button)findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Intent intent = new Intent("com.example.broadcasttest.MY_BROADCAST");//注意这里的值一定要和BroadCastTest中的一样。
                sendOrderedBroadcast(intent,null);//这里由sendBroadcast()方法变化。
            }
        });
    }

  ........
}

只是将sendBroadcast()方法改成sendOrderBroadcast()方法,sendOrderBroadcast()方法接收两个参数,第一个参数仍然是Intent,第二个参数是一个与权限相关的字符串,这里传入null就行了。重新运行程序,这两个应用程序仍然可以接收到这条广播。


但是这时候的广播接收器是有先后顺序的,而且前面的广播接收器还可以将广播截断,以阻止其传播。

那么该如何设定广播接收器的先后顺序呢?当然是在注册的时候进行设定的,修改AndroidManifest.xml中的代码:

如下:

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

    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name = "android.permision.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">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

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

        <receiver
            android:name=".MyBroadcastReceiver"
            android:enabled="true"
            android:exported="true">
            <intent-filter android:priority="100">//我们给广播接收器设置了优先级
                <action android:name="com.example.broadcasttest.MY_BROADCAST"/>
            </intent-filter>
        </receiver>
    </application>

</manifest>

可以看到我们通过android:priority属性给广播接收器设置了优先级,优先级高的广播接收器就可以先收到广播,这里将MyBroadcastReceiver的优先级设成100,以保证它一定会在AnotherBroadcastReceicer之前收到广播。

既然我们已经获得了接收广播的优先权,那么MyBroadCastReceiver就可以选择时候允许广播继续传递了。

修该MyBroadcastReceiver中的代码,如下:

package com.example.broadcasttest;

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

/**
 * Created by ZHJ on 2018/3/11.
 */

public class MyBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context,"received in MyBroadcastReceiver",Toast.LENGTH_SHORT).show();
        abortBroadcast();//调用abortBroadcast()方法。
    }
}

如果在onReceive()方法中调用了abortBroadcast()方法,就表示这条广播截断,优先级的广播接收器就无法收到这条广播。

重新运行程序。

只有MyBroadcastReceiver中的广播接收器的Toast信息框可以弹出。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值