广播分为两种类型:标准广播和有序广播
我们来看一下具体这两者的具体区别:
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信息框可以弹出。