热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

AndroidBroadcastReceiver学习

android,broadcastr
  BroadcastReceiver 用于异步接收广播Intent。主要有两大类,用于接收广播的:
  • 正常广播 Normal broadcasts(用 Context.sendBroadcast()发送)是完全异步的。它们都运行在一个未定义的顺序,通常是在同一时间。这样会更有效,但意味着receiver不能包含所要使用的结果或中止的API。  
  • 有序广播 Ordered broadcasts(用 Context.sendOrderedBroadcast()发送)每次被发送到一个receiver。所谓有序,就是每个receiver执行后可以传播到下一个receiver,也可以完全中止传播——不传播给其他receiver。 而receiver运行的顺序可以通过matched intent-filter 里面的android:priority来控制,当priority优先级相同的时候,Receiver以任意的顺序运行。

    要注意的是,即使是Normal broadcasts,系统在某些情况下可能会恢复到一次传播给一个receiver。 特别是receiver可能需要创建一个进程,为了避免系统超载,只能一次运行一个receiver。

    Broadcast Receiver 并没有提供可视化的界面来显示广播信息。可以使用Notification和Notification Manager来实现可视化的信息的界面,显示广播信息的内容,图标及震动信息。

生命周期
    一个BroadcastReceiver 对象只有在被调用 onReceive(Context, Intent)的才有效的,当从该函数返回后,该对象就无效的了,结束生命周期。
    因此从这个特征可以看出,在所调用的onReceive(Context, Intent)函数里,不能有过于耗时的操作,不能使用线程来执行。对于耗时的操作,请start service来完成。因为当得到其他异步操作所返回的结果时,BroadcastReceiver 可能已经无效了。

发送广播
    事件的广播比较简单,构建Intent对象,可调用sendBroadcast(Intent)方法将广播发出。另外还有sendOrderedBroadcast(),sendStickyBroadcast()等方法,请查阅API Doc。
    1.new Intent with action name 
        Intent intent = new Intent(String action);
      或者 只是new Intent, 然后
        intent.setAction(String action);

    2.set data等准备好了后,in activity,
        sendBroadcast(Intent); // 发送广播

接收广播
    通过定义一个继承BroadcastReceiver类来实现,继承该类后覆盖其onReceiver方法,并在该方法中响应事件。
public  class SMSReceiver  extends BroadcastReceiver { 

        @Override 
         public  void  onReceive(Context context, Intent intent) { 
                 // get data from SMS intent 
                Bundle bundle = intent.getExtras(); 
                 if (bundle !=  null){ 
                         // get message by "pdus" 
                        Object[] objArray = (Object[]) bundle.get( "pdus"); 

                         // rebuild SMS 
                        SmsMessage[] messages =  new SmsMessage[objArray.length]; 
                         for ( int i=0; i                                 messages[i] = SmsMessage.createFromPdu(( byte[])objArray[i]); 

                                StringBuilder str =  new StringBuilder( "from: "); 
                                str.append(messages[i].getDisplayOriginatingAddress()); 
                                str.append( "\nmessage:\n"); 
                                str.append(messages[i].getDisplayMessageBody()); 

                                Toast.makeText(context, str.toString(), Toast.LENGTH_LONG) 
                                                .show(); 
                        } 
                } 
        } 
}

注册Receiver
   注册有两种方式:
   1. 静态方式,在AndroidManifest.xml的application里面定义receiver并设置要接收的action。
< receiver  android:name =".SMSReceiver" > 
         < intent-filter > 
                 < action  android:name ="android.provider.Telephony.SMS_RECEIVED"  /> 
         intent-filter > 
receiver >

   2. 动态方式, 在activity里面调用函数来注册,和静态的内容差不多。一个形参是receiver,另一个是IntentFilter,其中里面是要接收的action。
public  class HelloDemo  extends Activity {    
         private BroadcastReceiver receiver;    

        @Override 
         protected  void onStart() { 
                 super.onStart(); 

                receiver =  new CallReceiver(); 
                registerReceiver(receiver,  new IntentFilter( "android.intent.action.PHONE_STATE")); 
        } 

        @Override 
         protected  void onStop() { 
                unregisterReceiver(receiver); 
                 super.onStop(); 
        } 
}
   一个receiver可以接收多个action的,即可以有多个intent-filter,需要在onReceive里面对intent.getAction(action name)进行判断。
 
    个人推荐使用静态注册方式,由系统来管理receiver,而且程序里的所有receiver,可以在xml里面一目了然。而动态注册方式,隐藏在代码中,比较难发现。
    而且动态注册,需要特别注意的是,在退出程序前要记得调用 Context.unregisterReceiver()方法。一般在activity的onStart()里面进行注册, onStop()里面进行注销。官方提醒,如果在Activity.onResume()里面注册了,就必须在Activity.onPause()注销。

Permission权限
  要接收某些action,需要在AndroidManifest.xml里面添加相应的permission。例如接收SMS:
< uses-permission  android:name ="android.permission.RECEIVE_SMS"  />


下面给出动态注册的接收来电的广播处理的CallReceiver的代码:
   一种方式是直接读取intent.getStringExtra("incoming_number")来获取来电号码:
public  class CallReceiver  extends BroadcastReceiver { 

        @Override 
         public  void onReceive(Context context, Intent intent) { 
                TelephonyManager teleManager = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE); 
                 
                 switch(teleManager.getCallState()){ 
                 case TelephonyManager.CALL_STATE_RINGING:  //响铃 
                        Toast.makeText(context,  "Ringing: " + intent.getStringExtra( "incoming_number"), Toast.LENGTH_LONG).show(); 
                         break
                 case TelephonyManager.CALL_STATE_OFFHOOK:  //接听 
                        Toast.makeText(context,  "OffHook: " + intent.getStringExtra( "incoming_number"), Toast.LENGTH_LONG).show(); 
                         break
                 case TelephonyManager.CALL_STATE_IDLE:  //挂断 
                        Toast.makeText(m_context,  "Idle: " + incomingNumber, Toast.LENGTH_LONG).show(); 
                         break
                } 
        } 
}
   在运行时,发现除了响铃时可以获取来电号码,接听和挂断都不能成功获取的,显示为null。

   另一种方式是通过PhoneStateListener的onCallStateChanged来监听状态的变化:
public  class CallReceiver  extends BroadcastReceiver { 

         private Context m_context; 
        @Override 
         public  void onReceive(Context context, Intent intent) { 
                m_cOntext= context; 
                TelephonyManager teleManager = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE); 
                 teleManager.listen(new PhoneStateListener()

                        @Override 
                         public  void onCallStateChanged( int state, String incomingNumber) { 
                                 switch(state){ 
                                 case TelephonyManager.CALL_STATE_RINGING:  //响铃 
                                        Toast.makeText(m_context,  "Ringing: " + incomingNumber, Toast.LENGTH_LONG) 
                                                                .show(); 
                                         break
                                 case TelephonyManager.CALL_STATE_OFFHOOK:  //接听 
                                        Toast.makeText(m_context,  "OffHook: " + incomingNumber, Toast.LENGTH_LONG) 
                                        .show(); 
                                         break
                                 case TelephonyManager.CALL_STATE_IDLE:  //挂断 
                                        Toast.makeText(m_context,  "Idle: " + incomingNumber, Toast.LENGTH_LONG) 
                                        .show(); 
                                         break
                                } 
                        }} , PhoneStateListener.LISTEN_CALL_STATE);  
        } 
}
    运行时也发现incomingNumber在接听和挂断时获取为blank。
    因为这里监听的是通话的状态变化,所以这个receiver会被调用3次。

    监听通话状态需要加上权限:
< uses-permission  android:name ="android.permission.READ_PHONE_STATE" />

===========
小结:
1. 对于sendBroadCast的intent对象,需要设置其action name; 
2. 推荐使用显式指明receiver,在配置文件AndroidManifest.xml指明; 
3. 一个receiver可以接收多个action; 
4. 每次接收广播都会重新生成一个接收广播的对象,再次调用onReceive;
5. 在BroadCast 中尽量不要处理太多逻辑问题,建议复杂的逻辑交给Activity 或者 Service 去处理。


本文转自 Icansoft 51CTO博客,原文链接:http://blog.51cto.com/android/521784

推荐阅读
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Imtryingtofigureoutawaytogeneratetorrentfilesfromabucket,usingtheAWSSDKforGo.我正 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • ZSI.generate.Wsdl2PythonError: unsupported local simpleType restriction ... [详细]
  • Android JSON基础,音视频开发进阶指南目录
    Array里面的对象数据是有序的,json字符串最外层是方括号的,方括号:[]解析jsonArray代码try{json字符串最外层是 ... [详细]
  • 本文详细介绍了Spring的JdbcTemplate的使用方法,包括执行存储过程、存储函数的call()方法,执行任何SQL语句的execute()方法,单个更新和批量更新的update()和batchUpdate()方法,以及单查和列表查询的query()和queryForXXX()方法。提供了经过测试的API供使用。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • 如何查询zone下的表的信息
    本文介绍了如何通过TcaplusDB知识库查询zone下的表的信息。包括请求地址、GET请求参数说明、返回参数说明等内容。通过curl方法发起请求,并提供了请求示例。 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • flowable工作流 流程变量_信也科技工作流平台的技术实践
    1背景随着公司业务发展及内部业务流程诉求的增长,目前信息化系统不能够很好满足期望,主要体现如下:目前OA流程引擎无法满足企业特定业务流程需求,且移动端体 ... [详细]
  • Go Cobra命令行工具入门教程
    本文介绍了Go语言实现的命令行工具Cobra的基本概念、安装方法和入门实践。Cobra被广泛应用于各种项目中,如Kubernetes、Hugo和Github CLI等。通过使用Cobra,我们可以快速创建命令行工具,适用于写测试脚本和各种服务的Admin CLI。文章还通过一个简单的demo演示了Cobra的使用方法。 ... [详细]
author-avatar
mobiledu2502922985
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有