博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
BroadcastReceiver
阅读量:6760 次
发布时间:2019-06-26

本文共 5617 字,大约阅读时间需要 18 分钟。

BroadcastReceiver

Android系统的四大组件还有一种BroadcastReceiver,这种组件本质上就是一种全局的监听器,用于监听系统全局的广播消息。由于BroadcastReceiver是一种全局的监听器,因此它可以非常方便地实现系统中不同组件之间的通信。例如我们希望客户端程序与startService()方法启动的Service之间通信,就可以借助于BroadcastReceiver来实现。

BroadcastReceiver简介

BroadcastReceiver用于接收程序(包括用户开发的程序和系统内建的程序)所发出的Broadcast Intent,与一块嗯用程序启动Activity、Service相同的是,程序启动BroadcastReceiver也只需要两步。

  1. 创建需要启动的BroadcastReceiver的Intent。
  2. 调用Context的sendBroadcast()或sendOrderedBroadcast()方法来启动指定的BroadcastReceiver。

当应用程序发出一个BroadcastReceiver Intent之后,所有匹配该Intent的BroadcastReceiver都有可能被启动。

与Activity、Service具有完整的生命周期不同,BroadcastReceiver本质上只是一个系统级的监听器——它专门负责监听个程序所发出的Broadcast。

由于BroadcastReceiver本质上属于一个监听器,因此实现BroadcastReceiver的方法也十分简单,只要重写BroadcastReceiver的onReceive(Context context, Intent intent)方法即可。

一旦实现了BroadcastReceiver,接下来就应该指定该BroadcastReceiver能匹配的Intent,此时有两种方式:

  • 使用代码进行指定,调用BroadcastReceiver的Context的registerReceiver(BroadcastReceiver receiver, IntentFilter filter)方法指定。例如如下代码:

    IntentFilter filter = new IntentFilter("android.provider.Telephony.SMS_RECEIVED");IncomingSMSReceiver receiver = new IncomingSMSReceiver();registerReceiver(receiver, filter);
  • 在AndroidManifest.xml文件中配置。例如如下代码:

每次系统Broadcast事件发生后,系统就会创建对应的BroadcastReceiver的实例,并自动触发它的onReceiver()方法,onReceiver()方法执行完后,BroadcastReceiver的实例就会被销毁。

如果BroadcastReceiver的方法不能在10秒内执行完成,Android会认为该程序无响应,所以不要在BroadcastReceiver的onReceiver()方法里执行―些耗时的操作,否则会弹出ANR(Application No Response)的对话框。

如果确实需要根据Broadcast来完成一项比较耗时的操作,则可以考虑通过Intent启动一个Service来完成该操作。不应考虑使用新线程去完成耗时的操作,因为BroadcastReceiver本身的生命周期很短,可能出现的情况是子线程可能还没有结束,BroadcastReceiver就已经退出了。

如果BroadcastReceiver所在的进程结束,虽然该进程内还有用户启动的新线程,但由于该进程内不包含任何活动组件,因此系统可能在内存紧张时优先结束该进程。这样就可能导致BroadcastReceiver启动的子线程不能执行完成。

发送广播

在程序中发送广播十分简单,只要调用Context的sendBroadcast(Intent intent)方法即可,这条广播将会启动intent参数所对应的BroadcastReceiver。

下面一个简单的程序示范了如何发送Broadcast、使用BroadcastReceiver接收广播,该程序的Activity界面中包含一个按钮,当用户单击该按钮时程序会向外发送一条广播。该程序的代码如下。

public class MainActivity extends Activity {
Button send; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); send = (Button) findViewById(R.id.send); send.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(); intent.setAction("tk.zongzhankui.broadcast.BROADCAST"); intent.putExtra("msg", "简单的消息"); sendBroadcast(intent); } }); }}
public class MyReceiver extends BroadcastReceiver {
public MyReceiver() { } @Override public void onReceive(Context context, Intent intent) { Toast.makeText(context, "接收到的Intent的Action为:" + intent.getAction() + "\n消息内容是:" + intent.getStringExtra("msg"), Toast.LENGTH_LONG).show(); }}

正如上面的程序中看到的,当符合该MyReceiver的广播出现时,该MyReceiver的onReceive()方法将会被触发,从而在该方法中显示广播所携带的消息。

上面发送广播的程序中指定发送广播时所用的Intent的Action为tk.zongzhankui.broadcast.BROADCAST,这就需要配置上面的BroadcastReceiver应监听Action为该字符串的Intent,在AndroidManifest.xml文件中增加如下配置即可:

有序广播

Broadcast被分为如下两种。

  • Normal Broadcast(普通广播):Normal Broadcast是完全异步的,可以在同一时刻(逻辑上)被所有接收者接收到,消息传递的效率比较高。但缺点是接收者不能将处理结果传递给下一个接收者,并且无法终止Broadcast Intent的传播。
  • Ordered Broadcast(有序广播):Ordered Broadcast的接收者将按预先声明的优先级依次接收Broadcast。如:A的级别高于B、B的级别高于C,那么,Broadcast先传给A,再传给B,最后传给C。优先级别声明在
public class MainActivity extends Activity {
Button send; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); send = (Button) findViewById(R.id.send); send.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(); intent.setAction("tk.zongzhankui.sortedbroadcast.BROADCAST"); intent.putExtra("msg", "简单的消息"); sendOrderedBroadcast(intent, null); } }); }}

上面的程序中粗体字代码指定了Intent的Action属性,再调用sendOrderedBroadcast()方法来发送有序广播。对于有序广播而言,它会按优先级依次出发每个BroadcastReceiver的onReceive()方法。

下面的程序先定义第一个BroadcastReceiver。

public class MyReceiver extends BroadcastReceiver {
public MyReceiver() { } @Override public void onReceive(Context context, Intent intent) { Toast.makeText(context, "接收到的Intent的Action为:" + intent.getAction() + "\n消息内容是:" + intent.getStringExtra("msg") , Toast.LENGTH_LONG).show(); Bundle bundle = new Bundle(); bundle.putString("first", "第一个BroadcastReceiver存入的消息"); setResultExtras(bundle);// abortBroadcast(); }}

上面的BroadcastReceiver不仅处理了它所接收到的消息,而且向处理结果中存入了key为frist的消息,这个消息将可以被第二个BroadcastReceiver解析出来。

在AndroidManifest.xml文件中部署该BroadcastReceiver,并指定其优先级为20,配置片段如下:

接下来再为程序提供第二个BroadcastReceiver,这个BroadcastReceiver将会解析前一个BroadcastReceiver所存入的key为first的消息。该BroadcastReceiver的代码如下。

public class MyReceiver2 extends BroadcastReceiver {
public MyReceiver2() { } @Override public void onReceive(Context context, Intent intent) { Bundle bundle = getResultExtras(true); String first = bundle.getString("first"); Toast.makeText(context, "第一个Broadcast存入的消息为:" + first, Toast.LENGTH_LONG).show(); }}

上面的程序中粗体字代码用于解析前一个BroadcastReceiver存入结果中key为first的消息,在AndroidManifest.xml文件中配置该BroadcastReceiver,并指定其优先级为0.配置片段如下:

转载于:https://www.cnblogs.com/zongzhankui/p/5875320.html

你可能感兴趣的文章
网络编程(一):用C#下载网络文件的2种方法
查看>>
复制graphic
查看>>
基于NopCommerce的开源电商系统改造总结
查看>>
JavaScript是怎样让互联网变慢的(及对策)
查看>>
诺基亚预装WIN8系统的LUMIA平板真机曝光-应用电台
查看>>
遇见C++ PPL:C++ 的并行和异步
查看>>
软件开发中关于向后兼容的理解
查看>>
ios开发之 MPMoviePlayerController 视频播放器
查看>>
count(*)、count(val)和count(1)的解释
查看>>
[Leetcode] Largest Rectangle in Histogram
查看>>
final (Java)
查看>>
The Master of Science degree in Computer Scienc
查看>>
利用Stack倒序List,利用Set使List不能添加重复元素
查看>>
移动云计算应用开发入门经典
查看>>
当跳票成了习惯
查看>>
InstallShield 10 Express 使用
查看>>
LightOJ 1230 Placing Lampposts(树形DP)
查看>>
rtems总结
查看>>
艾伟_转载:使用.NET框架自带的Json序列化类
查看>>
一起谈.NET技术,VS 2010 和 .NET 4.0 系列之《VS 2010代码智能提示的改进》篇
查看>>