广播机制

安卓的广播机制:

说明:广播广播,其实就是一个地方将信息传播出去,只要有广播接收器的地方都能同时接收到信息。安卓的广播机制一般可以针对应用内的广播,也可以针对跨应用广播。例如电池的事件,电池快没电的时候,会发送一个广播,如果你的应用中有接收这个广播,则就会调用广播接收的程序。

广播和Service都是在UI线程中处理的,所以使用广播的时候,不能处理延迟很长的操作,否则会阻碍UI线程的更新。通过广播,可以监听外部APP发送过来的广播消息,通过这些广播消息,可以做出相应的处理。

看一个图:


一些观点:

  1. 在Android 中如果要发送一个广播必须使用sendBroadCast 向系统发送对其感兴趣的广播接收器中
  2. 使用广播必须要有一个intent 对象必设置其action动作对象
  3. 使用广播必须在配置文件中显式的指明该广播对象
  4. 每次接收广播都会重新生成一个接收广播的对象
  5. 在BroadCast 中尽量不要处理太多逻辑问题,建议复杂的逻辑交给Activity 或者 Service 去处理


广播例子,常驻型广播

常驻型广播,当你的应用程序关闭了,如果有广播信息来,你写的广播接收器同样的能接受到,他的注册方式就是在你的应用程序中的AndroidManifast.xml进行注册。

首先需要配置AndroidMainfest.xml

[html]  view plain  copy
 print ?
  1. <!--  
  2.      android:name就是限定了只能在 TestReceiver   
  3.     广播接收器中接收 android.intent.action.EDIT类型的广播  
  4. -->  
  5. <receiver android:name=".TestReceiver" >  
  6.     <intent-filter>  
  7.         <action android:name="android.intent.action.EDIT" />  
  8.     </intent-filter>  
  9. </receiver>  
  10. <receiver android:name=".Test2Receiver" >  
  11.     <intent-filter>  
  12.         <action android:name="android.intent.action.EDIT" />  
  13.     </intent-filter>  
  14. </receiver>  


发送一个广播 MainActivity.java:

[java]  view plain  copy
 print ?
  1. package com.example.test.com;  
  2.   
  3.   
  4. import android.annotation.SuppressLint;  
  5. import android.app.Activity;  
  6. import android.content.Intent;  
  7. import android.os.Bundle;  
  8. import android.view.View;  
  9. import android.view.View.OnClickListener;  
  10. import android.view.Window;  
  11. import android.widget.Button;  
  12.   
  13. @SuppressLint("HandlerLeak")  
  14. public class MainActivity extends Activity {  
  15.   
  16.     private Button btn;  
  17.   
  18.     @Override  
  19.     protected void onCreate(Bundle savedInstanceState) {  
  20.         super.onCreate(savedInstanceState);  
  21.         requestWindowFeature(Window.FEATURE_NO_TITLE);  
  22.         setContentView(R.layout.activity_main);  
  23.   
  24.         btn = (Button) findViewById(R.id.button1);  
  25.         //图片点击的时候  
  26.         btn.setOnClickListener(new OnClickListener() {  
  27.   
  28.             @Override  
  29.             public void onClick(View v) {  
  30.                 Intent intent = new Intent();//采用intent发送数据  
  31.                 intent.setAction(Intent.ACTION_EDIT);//只发送给action为ACTION_EDIT的对象  
  32.                 Bundle bundle = new Bundle();  
  33.                 bundle.putString("val""Hello World");  
  34.                 intent.putExtras(bundle);  
  35.                 //通过广播方式发送该intent,只有特定action的接收者才能收到该广播信息  
  36.                 MainActivity.this.sendBroadcast(intent);  
  37.             }  
  38.         });  
  39.     }  
  40.   
  41. }  

接收广播,需要继承BroadcastReceiver类,并且实现onReceive方法:

TestReceiver:

[java]  view plain  copy
 print ?
  1. package com.example.test.com;  
  2.   
  3.   
  4. import android.content.BroadcastReceiver;  
  5. import android.content.Context;  
  6. import android.content.Intent;  
  7. import android.os.Bundle;  
  8.   
  9. public class TestReceiver extends BroadcastReceiver {  
  10.   
  11.     public TestReceiver() {  
  12.         System.out.println("TestReceiver");  
  13.   
  14.     }  
  15.   
  16.     @Override  
  17.     public void onReceive(Context context, Intent intent) {  
  18.         String action = intent.getAction();  
  19.         //这边可以获取Action的名称  
  20.         if (action.equals(Intent.ACTION_EDIT)) {  
  21.             Bundle bundle = intent.getExtras();  
  22.             System.out.println("OnReceiver" + bundle.getString("val"));  
  23.         }  
  24.     }  
  25. }  


Test2Receiver:

[java]  view plain  copy
 print ?
  1. package com.example.test.com;  
  2.   
  3.   
  4. import android.content.BroadcastReceiver;  
  5. import android.content.Context;  
  6. import android.content.Intent;  
  7. import android.os.Bundle;  
  8.   
  9. public class Test2Receiver extends BroadcastReceiver {  
  10.   
  11.     public Test2Receiver() {  
  12.         System.out.println("Test2Receiver");  
  13.   
  14.     }  
  15.   
  16.     @Override  
  17.     public void onReceive(Context context, Intent intent) {  
  18.         String action = intent.getAction();  
  19.         if (action.equals(Intent.ACTION_EDIT)) {  
  20.             Bundle bundle = intent.getExtras();  
  21.             System.out.println("OnReceiver2" + bundle.getString("val"));  
  22.         }  
  23.     }  
  24. }  

点击按钮后,控制台效果:


接收器的生命周期:



广播例子,非常驻型广播

当应用程序结束了,广播自然就没有了,比如你在activity中的onCreate或者onResume中注册广播接收器在onDestory中卸载广播接收器。这样你的广播接收器就一个非常驻型的了。这种也叫动态注册。

[java]  view plain  copy
 print ?
  1. package com.example.test.com;  
  2.   
  3.   
  4. import android.annotation.SuppressLint;  
  5. import android.app.Activity;  
  6. import android.content.Intent;  
  7. import android.content.IntentFilter;  
  8. import android.os.Bundle;  
  9. import android.view.View;  
  10. import android.view.View.OnClickListener;  
  11. import android.view.Window;  
  12. import android.widget.Button;  
  13.   
  14. @SuppressLint("HandlerLeak")  
  15. public class MainActivity extends Activity {  
  16.   
  17.     private Button       btn;  
  18.     private TestReceiver testReceiver;  
  19.   
  20.     @Override  
  21.     protected void onCreate(Bundle savedInstanceState) {  
  22.         super.onCreate(savedInstanceState);  
  23.         requestWindowFeature(Window.FEATURE_NO_TITLE);  
  24.         setContentView(R.layout.activity_main);  
  25.   
  26.         //初始化的时候注册  
  27.         IntentFilter filter = new IntentFilter();  
  28.         filter.addAction(Intent.ACTION_EDIT);  
  29.         testReceiver = new TestReceiver();  
  30.         registerReceiver(testReceiver, filter);  
  31.   
  32.         btn = (Button) findViewById(R.id.button1);   
  33.         btn.setOnClickListener(new OnClickListener() {  
  34.   
  35.             @Override  
  36.             public void onClick(View v) {  
  37.                 Intent intent = new Intent();//采用intent发送数据    
  38.                 intent.setAction(Intent.ACTION_EDIT);//只发送给action为ACTION_EDIT的对象    
  39.                 Bundle bundle = new Bundle();    
  40.                 bundle.putString("val""Hello World");    
  41.                 intent.putExtras(bundle);    
  42.                 //通过广播方式发送该intent,只有特定action的接收者才能收到该广播信息    
  43.                 MainActivity.this.sendBroadcast(intent);    
  44.   
  45.             }  
  46.   
  47.         });  
  48.     }  
  49.   
  50.     @Override  
  51.     protected void onDestroy() {  
  52.         unregisterReceiver(testReceiver);  
  53.     }  
  54.   
  55. }  

[java]  view plain  copy
 print ?
  1. package com.example.test.com;  
  2.   
  3.   
  4. import android.content.BroadcastReceiver;  
  5. import android.content.Context;  
  6. import android.content.Intent;  
  7. import android.os.Bundle;  
  8.   
  9. public class TestReceiver extends BroadcastReceiver {  
  10.   
  11.     public TestReceiver() {  
  12.         System.out.println("TestReceiver");  
  13.   
  14.     }  
  15.   
  16.     @Override  
  17.     public void onReceive(Context context, Intent intent) {  
  18.         String action = intent.getAction();  
  19.         //这边可以获取Action的名称    
  20.         if (action.equals(Intent.ACTION_EDIT)) {  
  21.             Bundle bundle = intent.getExtras();  
  22.             System.out.println("OnReceiver" + bundle.getString("val"));  
  23.         }  
  24.     }  
  25. }  

下面是一个内部类实现广播接收的例子,通过内部类实现,可以动态监听广播消息,并且对广播进行UI界面上的处理

[java]  view plain  copy
 print ?
  1. package com.example.test.com;  
  2.   
  3.   
  4. import android.app.Activity;  
  5. import android.content.BroadcastReceiver;  
  6. import android.content.Context;  
  7. import android.content.Intent;  
  8. import android.content.IntentFilter;  
  9. import android.os.Bundle;  
  10. import android.view.View;  
  11. import android.view.View.OnClickListener;  
  12. import android.view.Window;  
  13. import android.widget.Button;  
  14. import android.widget.Toast;  
  15.   
  16. public class MainActivity extends Activity {  
  17.   
  18.     private ButtonOnClickListener btnListener;  
  19.   
  20.     private Button btn;  
  21.   
  22.     @Override  
  23.     protected void onCreate(Bundle savedInstanceState) {  
  24.         super.onCreate(savedInstanceState);  
  25.         requestWindowFeature(Window.FEATURE_NO_TITLE);  
  26.         setContentView(R.layout.main);  
  27.         btn = (Button) findViewById(R.id.button1);  
  28.         btnListener = new ButtonOnClickListener();  
  29.         btn.setOnClickListener(btnListener);  
  30.   
  31.     }  
  32.   
  33.     class ButtonOnClickListener implements OnClickListener {  
  34.   
  35.         @Override  
  36.         public void onClick(View v) {  
  37.             Toast.makeText(getBaseContext(), "YES!", Toast.LENGTH_SHORT).show();  
  38.               
  39.             TestBroadcast testBroadCast = new TestBroadcast();  
  40.             IntentFilter filter = new IntentFilter();  
  41.             filter.addAction(Intent.ACTION_EDIT);    //只有持有相同的action的接受者才能接收此广播  
  42.             registerReceiver(testBroadCast, filter); //注册  
  43.               
  44.             Intent intent = new Intent();//采用intent发送数据      
  45.             Bundle bundle = new Bundle();  
  46.             bundle.putString("hello""world");  
  47.             intent.putExtras(bundle);  
  48.             intent.setAction(Intent.ACTION_EDIT);//只发送给action为ACTION_EDIT的对象      
  49.             MainActivity.this.sendBroadcast(intent);    
  50.         }  
  51.     }  
  52.   
  53.     //通过内部类,可以实现UI修改  
  54.     class TestBroadcast extends BroadcastReceiver {  
  55.   
  56.         @Override  
  57.         public void onReceive(Context context, Intent intent) {  
  58.             Toast.makeText(context, "YES2!", Toast.LENGTH_SHORT).show();  
  59.             String action = intent.getAction();  
  60.             if (action.equals(Intent.ACTION_EDIT)) {  
  61.                 Bundle bundle = intent.getExtras();  
  62.                 btn.setText(bundle.getString("hello")); //更改btn的文字  
  63.             }  
  64.         }  
  65.     }  
  66.   
  67.   
  68.   
  69. }  


安卓系统广播

[java]  view plain  copy
 print ?
  1. String ADD_SHORTCUT_ACTION 动作:在系统中添加一个快捷方式。  
  2. String ALL_APPS_ACTION 动作:列举所有可用的应用。输入:无。  
  3. String ALTERNATIVE_CATEGORY 类别:说明 activity 是用户正在浏览的数据的一个可选操作。  
  4. String ANSWER_ACTION 动作:处理拨入的电话。  
  5. String BATTERY_CHANGED_ACTION 广播:充电状态,或者电池的电量发生变化。  
  6. String BOOT_COMPLETED_ACTION 广播:在系统启动后,这个动作被广播一次(只有一次)。  
  7. String BROWSABLE_CATEGORY 类别:能够被浏览器安全使用的 activities 必须支持这个类别。  
  8. String BUG_REPORT_ACTION 动作:显示 activity 报告错误。  
  9. String CALL_ACTION 动作:拨打电话,被呼叫的联系人在数据中指定。  
  10. String CALL_FORWARDING_STATE_CHANGED_ACTION 广播:语音电话的呼叫转移状态已经改变。  
  11. String CLEAR_CREDENTIALS_ACTION 动作:清除登陆凭证 (credential)。  
  12. String CONFIGURATION_CHANGED_ACTION 广播:设备的配置信息已经改变,参见 Resources.Configuration.  
  13. Creator CREATOR 无 无  
  14. String DATA_ACTIVITY_STATE_CHANGED_ACTION 广播:电话的数据活动(data activity)状态(即收发数据的状态)已经改变。  
  15. String DATA_CONNECTION_STATE_CHANGED_ACTION 广播:电话的数据连接状态已经改变。  
  16. String DATE_CHANGED_ACTION 广播:日期被改变。  
  17. String DEFAULT_ACTION 动作:和 VIEW_ACTION 相同,是在数据上执行的标准动作。  
  18. String DEFAULT_CATEGORY 类别:如果 activity 是对数据执行确省动作(点击, center press)的一个选项,需要设置这个类别。  
  19. String DELETE_ACTION 动作:从容器中删除给定的数据。  
  20. String DEVELOPMENT_PREFERENCE_CATEGORY 类别:说明 activity 是一个设置面板 (development preference panel).  
  21. String DIAL_ACTION 动作:拨打数据中指定的电话号码。  
  22. String EDIT_ACTION 动作:为制定的数据显示可编辑界面。  
  23. String EMBED_CATEGORY 类别:能够在上级(父)activity 中运行。  
  24. String EMERGENCY_DIAL_ACTION 动作:拨打紧急电话号码。  
  25. int FORWARD_RESULT_LAUNCH 启动标记:如果这个标记被设置,而且被一个已经存在的 activity 用来启动新的 activity,已有 activity 的回复目标 (reply target) 会被转移给新的 activity。  
  26. String FOTA_CANCEL_ACTION 广播:取消所有被挂起的 (pending) 更新下载。  
  27. String FOTA_INSTALL_ACTION 广播:更新已经被确认,马上就要开始安装。  
  28. String FOTA_READY_ACTION 广播:更新已经被下载,可以开始安装。  
  29. String FOTA_RESTART_ACTION 广播:恢复已经停止的更新下载。  
  30. String FOTA_UPDATE_ACTION 广播:通过 OTA 下载并安装操作系统更新。  
  31. String FRAMEWORK_INSTRUMENTATION_TEST_CATEGORY 类别:To be used as code under test for framework instrumentation tests.  
  32. String GADGET_CATEGORY 类别:这个 activity 可以被嵌入宿主 activity (activity that is hosting gadgets)。  
  33. String GET_CONTENT_ACTION 动作:让用户选择数据并返回。  
  34. String HOME_CATEGORY 类别:主屏幕 (activity),设备启动后显示的第一个 activity。  
  35. String INSERT_ACTION 动作:在容器中插入一个空项 (item)。  
  36. String INTENT_EXTRA 附加数据:和 PICK_ACTIVITY_ACTION 一起使用时,说明用户选择的用来显示的 activity;和 ADD_SHORTCUT_ACTION 一起使用的时候,描述要添加的快捷方式。  
  37. String LABEL_EXTRA 附加数据:大写字母开头的字符标签,和 ADD_SHORTCUT_ACTION 一起使用。  
  38. String LAUNCHER_CATEGORY 类别:Activity 应该被显示在顶级的 launcher 中。  
  39. String LOGIN_ACTION 动作:获取登录凭证。  
  40. String MAIN_ACTION 动作:作为主入口点启动,不需要数据。  
  41. String MEDIABUTTON_ACTION 广播:用户按下了“Media Button”。  
  42. String MEDIA_BAD_REMOVAL_ACTION 广播:扩展介质(扩展卡)已经从 SD 卡插槽拔出,但是挂载点 (mount point) 还没解除 (unmount)。  
  43. String MEDIA_EJECT_ACTION 广播:用户想要移除扩展介质(拔掉扩展卡)。  
  44. String MEDIA_MOUNTED_ACTION 广播:扩展介质被插入,而且已经被挂载。  
  45. String MEDIA_REMOVED_ACTION 广播:扩展介质被移除。  
  46. String MEDIA_SCANNER_FINISHED_ACTION 广播:已经扫描完介质的一个目录。  
  47. String MEDIA_SCANNER_STARTED_ACTION 广播:开始扫描介质的一个目录。  
  48. String MEDIA_SHARED_ACTION 广播:扩展介质的挂载被解除 (unmount),因为它已经作为 USB 大容量存储被共享。  
  49. String MEDIA_UNMOUNTED_ACTION 广播:扩展介质存在,但是还没有被挂载 (mount)。  
  50. String MESSAGE_WAITING_STATE_CHANGED_ACTION 广播:电话的消息等待(语音邮件)状态已经改变。  
  51. int MULTIPLE_TASK_LAUNCH 启动标记:和 NEW_TASK_LAUNCH 联合使用,禁止将已有的任务改变为前景任务 (foreground)。  
  52. String NETWORK_TICKLE_RECEIVED_ACTION 广播:设备收到了新的网络 "tickle" 通知。  
  53. int NEW_TASK_LAUNCH 启动标记:设置以后,activity 将成为历史堆栈中的第一个新任务(栈顶)。  
  54. int NO_HISTORY_LAUNCH 启动标记:设置以后,新的 activity 不会被保存在历史堆栈中。  
  55. String PACKAGE_ADDED_ACTION 广播:设备上新安装了一个应用程序包。  
  56. String PACKAGE_REMOVED_ACTION 广播:设备上删除了一个应用程序包。  
  57. String PHONE_STATE_CHANGED_ACTION 广播:电话状态已经改变。  
  58. String PICK_ACTION 动作:从数据中选择一个项目 (item),将被选中的项目返回。  
  59. String PICK_ACTIVITY_ACTION 动作:选择一个 activity,返回被选择的 activity 的类(名)。  
  60. String PREFERENCE_CATEGORY 类别:activity是一个设置面板 (preference panel)。  
  61. String PROVIDER_CHANGED_ACTION 广播:更新将要(真正)被安装。  
  62. String PROVISIONING_CHECK_ACTION 广播:要求 polling of provisioning service 下载最新的设置。  
  63. String RUN_ACTION 动作:运行数据(指定的应用),无论它(应用)是什么。  
  64. String SAMPLE_CODE_CATEGORY 类别:To be used as an sample code example (not part of the normal user experience).  
  65. String SCREEN_OFF_ACTION 广播:屏幕被关闭。  
  66. String SCREEN_ON_ACTION 广播:屏幕已经被打开。  
  67. String SELECTED_ALTERNATIVE_CATEGORY 类别:对于被用户选中的数据,activity 是它的一个可选操作。  
  68. String SENDTO_ACTION 动作:向 data 指定的接收者发送一个消息。  
  69. String SERVICE_STATE_CHANGED_ACTION 广播:电话服务的状态已经改变。  
  70. String SETTINGS_ACTION 动作:显示系统设置。输入:无。  
  71. String SIGNAL_STRENGTH_CHANGED_ACTION 广播:电话的信号强度已经改变。  
  72. int SINGLE_TOP_LAUNCH 启动标记:设置以后,如果 activity 已经启动,而且位于历史堆栈的顶端,将不再启动(不重新启动) activity。  
  73. String STATISTICS_REPORT_ACTION 广播:要求 receivers 报告自己的统计信息。  
  74. String STATISTICS_STATE_CHANGED_ACTION 广播:统计信息服务的状态已经改变。  
  75. String SYNC_ACTION 动作:执行数据同步。  
  76. String TAB_CATEGORY 类别:这个 activity 应该在 TabActivity 中作为一个 tab 使用。  
  77. String TEMPLATE_EXTRA 附加数据:新记录的初始化模板。  
  78. String TEST_CATEGORY 类别:作为测试目的使用,不是正常的用户体验的一部分。  
  79. String TIMEZONE_CHANGED_ACTION 广播:时区已经改变。  
  80. String TIME_CHANGED_ACTION 广播:时间已经改变(重新设置)。  
  81. String TIME_TICK_ACTION 广播:当前时间已经变化(正常的时间流逝)。  
  82. String UMS_CONNECTED_ACTION 广播:设备进入 USB 大容量存储模式。  
  83. String UMS_DISCONNECTED_ACTION 广播:设备从 USB 大容量存储模式退出。  
  84. String UNIT_TEST_CATEGORY 类别:应该被用作单元测试(通过 test harness 运行)。  
  85. String VIEW_ACTION 动作:向用户显示数据。  
  86. String WALLPAPER_CATEGORY 类别:这个 activity 能过为设备设置墙纸。  
  87. String WALLPAPER_CHANGED_ACTION 广播:系统的墙纸已经改变。  
  88. String WALLPAPER_SETTINGS_ACTION 动作:显示选择墙纸的设置界面。输入:无。  
  89. String WEB_SEARCH_ACTION 动作:执行 web 搜索。  
  90. String XMPP_CONNECTED_ACTION 广播:XMPP 连接已经被建立。  
  91. String XMPP_DISCONNECTED_ACTION 广播:XMPP 连接已经被断开。  
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
NumPy广播机制是一种用于在不同形状的数组之间进行元素级操作的机制。它允许具有不同形状的数组进行算术运算,而无需显式地扩展数组的形状。广播机制通过在缺失或大小为1的维度上扩展数组来实现这一点,以使其具有兼容的形状。 广播机制遵循以下规则: 1. 如果两个数组的维度数量不同,则在维度较小的数组前面补1,直到两个数组的维度数量相同。 2. 如果两个数组在某个维度上的大小不同,并且其中一个数组在该维度上的大小为1,则可以沿着该维度进行广播。 3. 如果两个数组在某个维度上的大小不相等,并且两个数组在该维度上的大小都不为1,则无法进行广播引发错误。 以下是一个示例,展示了广播机制应用: ```python import numpy as np a = np.array([[0, 0, 0], [10, 10, 10], [20, 20, 20], [30, 30, 30]]) b = np.array([1, 2, 3]) print(a * b) ``` 在这个例子中,数组`a`的形状是(4, 3),数组`b`的形状是(3,),但是它们可以进行乘法运算,因为在第二个维度上,数组`b`的大小为1,可以通过广播机制自动扩展为(4, 3)的形状,以与数组`a`相匹配。 输出结果为: ``` array([[ 0, 0, 0], [10, 20, 30], [20, 40, 60], [30, 60, 90]]) ``` 这是通过将数组`b`扩展为与数组`a`相同的形状,然后进行元素级乘法运算得到的结果。请注意,广播机制进行运算时,并不实际复制数组的,而是利用广播的原理进行计算,从而提高了效率。 总之,NumPy广播机制是一种非常强大的工具,可以方便地执行在不同形状的数组上进行元素级操作的任务。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值