加载中...

4.3.2 BroadcastReceiver庖丁解牛


本节引言:

上节我们对BroadcastReceiver已经有了一个初步的了解了,知道两种广播类型:标准与有序, 动态或静态注册广播接收者,监听系统广播,自己发送广播!已经满足我们的基本需求了~ 但是前面写的广播都是全局广播!这同样意味着我们APP发出的广播,其他APP都会接收到, 或者其他APP发送的广播,我们的APP也同样会接收到,这样容易引起一些安全性的问题!而 Android中给我们提供了本地广播的机制,使用该机制发出的广播只会在APP内部传播,而且 广播接收者也只能收到本应用发出的广播!

1.本地广播

1)核心用法:

PS:本地广播无法通过静态注册方式来接受,相比起系统全局广播更加高效

2)注意事项:

3)代码示例(别处登陆踢用户下线):

像微信一样,正在运行的微信,如果我们用别的手机再次登陆自己的账号,前面这个是会提醒账户 在别的终端登录这样,然后把我们打开的所有Activity都关掉,然后回到登陆页面这样~
下面我们就来写个简单的例子:

运行效果图:

代码实现:

Step 1:准备一个关闭所有Activity的ActivityCollector ,这里之前用前面Activity提供的那个!

ActivityCollector.java

  1. public class ActivityCollector {
  2. private static List<Activity> activities = new ArrayList<Activity>();
  3. public static void addActivity(Activity activity) {
  4. activities.add(activity);
  5. }
  6. public static void removeActivity(Activity activity) {
  7. activities.remove(activity);
  8. }
  9. public static void finishAll() {
  10. for (Activity activity : activities) {
  11. if (!activity.isFinishing()) {
  12. activity.finish();
  13. }
  14. }
  15. }
  16. }

Step 2:先写要给简单的BaseActivity,用来继承,接着写下登陆界面!

  1. public class BaseActivity extends AppCompatActivity {
  2. @Override
  3. protected void onCreate(Bundle savedInstanceState) {
  4. super.onCreate(savedInstanceState);
  5. ActivityCollector.addActivity(this);
  6. }
  7.  
  8. @Override
  9. protected void onDestroy() {
  10. super.onDestroy();
  11. ActivityCollector.removeActivity(this);
  12. }
  13. }

LoginActivity.java:

  1. public class LoginActivity extends BaseActivity implements View.OnClickListener{
  2.  
  3.  
  4. private SharedPreferences pref;
  5. private SharedPreferences.Editor editor;
  6.  
  7. private EditText edit_user;
  8. private EditText edit_pawd;
  9. private Button btn_login;
  10.  
  11.  
  12. @Override
  13. protected void onCreate(Bundle savedInstanceState) {
  14. super.onCreate(savedInstanceState);
  15. setContentView(R.layout.activity_login);
  16. pref = PreferenceManager.getDefaultSharedPreferences(this);
  17.  
  18. bindViews();
  19. }
  20.  
  21. private void bindViews() {
  22. edit_user = (EditText) findViewById(R.id.edit_user);
  23. edit_pawd = (EditText) findViewById(R.id.edit_pawd);
  24. btn_login = (Button) findViewById(R.id.btn_login);
  25. btn_login.setOnClickListener(this);
  26. }
  27.  
  28. @Override
  29. protected void onStart() {
  30. super.onStart();
  31. if(!pref.getString("user","").equals("")){
  32. edit_user.setText(pref.getString("user",""));
  33. edit_pawd.setText(pref.getString("pawd",""));
  34. }
  35. }
  36.  
  37. @Override
  38. public void onClick(View v) {
  39. String user = edit_user.getText().toString();
  40. String pawd = edit_pawd.getText().toString();
  41. if(user.equals("123")&&pawd.equals("123")){
  42. editor = pref.edit();
  43. editor.putString("user", user);
  44. editor.putString("pawd", pawd);
  45. editor.commit();
  46. Intent intent = new Intent(LoginActivity.this, MainActivity.class);
  47. startActivity(intent);
  48. Toast.makeText(LoginActivity.this,"哟,竟然蒙对了~",Toast.LENGTH_SHORT).show();
  49. finish();
  50. }else{
  51. Toast.makeText(LoginActivity.this,"这么简单都输出,脑子呢?",Toast.LENGTH_SHORT).show();
  52. }
  53.  
  54. }
  55. }

Step 3:自定义一个BroadcastReceiver,在onReceive里完成弹出对话框操作,以及启动登陆页面: MyBcReceiver.java

  1. public class MyBcReceiver extends BroadcastReceiver {
  2. @Override
  3. public void onReceive(final Context context, Intent intent) {
  4. AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(context);
  5. dialogBuilder.setTitle("警告:");
  6. dialogBuilder.setMessage("您的账号在别处登录,请重新登陆~");
  7. dialogBuilder.setCancelable(false);
  8. dialogBuilder.setPositiveButton("确定",
  9. new DialogInterface.OnClickListener() {
  10. @Override
  11. public void onClick(DialogInterface dialog, int which) {
  12. ActivityCollector.finishAll();
  13. Intent intent = new Intent(context, LoginActivity.class);
  14. intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  15. context.startActivity(intent);
  16. }
  17. });
  18. AlertDialog alertDialog = dialogBuilder.create();
  19. alertDialog.getWindow().setType(
  20. WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
  21. alertDialog.show();
  22. }
  23. }

别忘了AndroidManifest.xml中加上系统对话框权限: < uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />

Step 4:在MainActivity中,实例化localBroadcastManager,拿他完成相关操作,另外销毁时 注意unregisterReceiver!

MainActivity.java

  1. public class MainActivity extends BaseActivity {
  2.  
  3. private MyBcReceiver localReceiver;
  4. private LocalBroadcastManager localBroadcastManager;
  5. private IntentFilter intentFilter;
  6.  
  7. @Override
  8. protected void onCreate(Bundle savedInstanceState) {
  9. super.onCreate(savedInstanceState);
  10. setContentView(R.layout.activity_main);
  11. localBroadcastManager = LocalBroadcastManager.getInstance(this);
  12.  
  13. //初始化广播接收者,设置过滤器
  14. localReceiver = new MyBcReceiver();
  15. intentFilter = new IntentFilter();
  16. intentFilter.addAction("com.jay.mybcreceiver.LOGIN_OTHER");
  17. localBroadcastManager.registerReceiver(localReceiver, intentFilter);
  18.  
  19. Button btn_send = (Button) findViewById(R.id.btn_send);
  20. btn_send.setOnClickListener(new View.OnClickListener() {
  21. @Override
  22. public void onClick(View v) {
  23. Intent intent = new Intent("com.jay.mybcreceiver.LOGIN_OTHER");
  24. localBroadcastManager.sendBroadcast(intent);
  25. }
  26. });
  27. }
  28.  
  29. @Override
  30. protected void onDestroy() {
  31. super.onDestroy();
  32. localBroadcastManager.unregisterReceiver(localReceiver);
  33. }
  34. }

好的,就是这么简单,别忘记注册Activity哦~

2.Android 4.3以上版本监听开机启动广播的问题解决:

在Android 4.3以上的版本,允许我们将应用安装在SD上,我们都知道是系统开机 间隔一小段时间后,才装载SD卡的,这样我们的应用就可能监听不到这个广播了! 所以我们需要既监听开机广播又监听SD卡挂载广播!

另外,有些手机可能并没有SD卡,所以这两个广播监听我们不能写到同一个Intetn-filter里 而是应该写成两个,配置代码如下:

  1. <receiver android:name=".MyBroadcastReceiver">
  2. <intent-filter>
  3. <action android:name="android.intent.action.BOOT_COMPLETED"/>
  4. </intent-filter>
  5. <intent-filter>
  6. <action android:name="ANDROID.INTENT.ACTION.MEDIA_MOUNTED"/>
  7. <action android:name="ANDROID.INTENT.ACTION.MEDIA_UNMOUNTED"/>
  8. <data android:scheme="file"/>
  9. </intent-filter>
  10. </receiver>

3.常用的系统广播总结:

最后给大家提供下我们平常可能会用到的一些系统广播吧:

  1. Intent.ACTION_AIRPLANE_MODE_CHANGED;
  2. //关闭或打开飞行模式时的广播
  3.  
  4. <strong>Intent.ACTION_BATTERY_CHANGED;
  5. //充电状态,或者电池的电量发生变化
  6. //电池的充电状态、电荷级别改变,不能通过组建声明接收这个广播,只有通过Context.registerReceiver()注册
  7.  
  8. <strong>Intent.ACTION_BATTERY_LOW;
  9. //表示电池电量低
  10.  
  11. <strong>Intent.ACTION_BATTERY_OKAY;
  12. //表示电池电量充足,即从电池电量低变化到饱满时会发出广播
  13.  
  14. Intent.ACTION_BOOT_COMPLETED;
  15. //在系统启动完成后,这个动作被广播一次(只有一次)。
  16.  
  17. Intent.ACTION_CAMERA_BUTTON;
  18. //按下照相时的拍照按键(硬件按键)时发出的广播
  19.  
  20. Intent.ACTION_CLOSE_SYSTEM_DIALOGS;
  21. //当屏幕超时进行锁屏时,当用户按下电源按钮,长按或短按(不管有没跳出话框),进行锁屏时,android系统都会广播此Action消息
  22.  
  23. Intent.ACTION_CONFIGURATION_CHANGED;
  24. //设备当前设置被改变时发出的广播(包括的改变:界面语言,设备方向,等,请参考Configuration.java)
  25.  
  26. Intent.ACTION_DATE_CHANGED;
  27. //设备日期发生改变时会发出此广播
  28.  
  29. Intent.ACTION_DEVICE_STORAGE_LOW;
  30. //设备内存不足时发出的广播,此广播只能由系统使用,其它APP不可用?
  31.  
  32. Intent.ACTION_DEVICE_STORAGE_OK;
  33. //设备内存从不足到充足时发出的广播,此广播只能由系统使用,其它APP不可用?
  34.  
  35. Intent.ACTION_DOCK_EVENT;
  36. //
  37. //发出此广播的地方frameworks\base\services\java\com\android\server\DockObserver.java
  38.  
  39. Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE;
  40. ////移动APP完成之后,发出的广播(移动是指:APP2SD)
  41.  
  42. Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE;
  43. //正在移动APP时,发出的广播(移动是指:APP2SD)
  44.  
  45. Intent.ACTION_GTALK_SERVICE_CONNECTED;
  46. //Gtalk已建立连接时发出的广播
  47.  
  48. Intent.ACTION_GTALK_SERVICE_DISCONNECTED;
  49. //Gtalk已断开连接时发出的广播
  50.  
  51. Intent.ACTION_HEADSET_PLUG;
  52. //在耳机口上插入耳机时发出的广播
  53.  
  54. Intent.ACTION_INPUT_METHOD_CHANGED;
  55. //改变输入法时发出的广播
  56.  
  57. Intent.ACTION_LOCALE_CHANGED;
  58. //设备当前区域设置已更改时发出的广播
  59.  
  60. Intent.ACTION_MANAGE_PACKAGE_STORAGE;
  61. //
  62.  
  63. Intent.ACTION_MEDIA_BAD_REMOVAL;
  64. //未正确移除SD卡(正确移除SD卡的方法:设置--SD卡和设备内存--卸载SD卡),但已把SD卡取出来时发出的广播
  65. //广播:扩展介质(扩展卡)已经从 SD 卡插槽拔出,但是挂载点 (mount point) 还没解除 (unmount)
  66.  
  67. Intent.ACTION_MEDIA_BUTTON;
  68. //按下"Media Button" 按键时发出的广播,假如有"Media Button" 按键的话(硬件按键)
  69.  
  70. Intent.ACTION_MEDIA_CHECKING;
  71. //插入外部储存装置,比如SD卡时,系统会检验SD卡,此时发出的广播?
  72. Intent.ACTION_MEDIA_EJECT;
  73. //已拔掉外部大容量储存设备发出的广播(比如SD卡,或移动硬盘),不管有没有正确卸载都会发出此广播?
  74. //广播:用户想要移除扩展介质(拔掉扩展卡)。
  75. Intent.ACTION_MEDIA_MOUNTED;
  76. //插入SD卡并且已正确安装(识别)时发出的广播
  77. //广播:扩展介质被插入,而且已经被挂载。
  78. Intent.ACTION_MEDIA_NOFS;
  79. //
  80. Intent.ACTION_MEDIA_REMOVED;
  81. //外部储存设备已被移除,不管有没正确卸载,都会发出此广播?
  82. // 广播:扩展介质被移除。
  83. Intent.ACTION_MEDIA_SCANNER_FINISHED;
  84. //广播:已经扫描完介质的一个目录
  85. Intent.ACTION_MEDIA_SCANNER_SCAN_FILE;
  86. //
  87. Intent.ACTION_MEDIA_SCANNER_STARTED;
  88. //广播:开始扫描介质的一个目录
  89.  
  90. Intent.ACTION_MEDIA_SHARED;
  91. // 广播:扩展介质的挂载被解除 (unmount),因为它已经作为 USB 大容量存储被共享。
  92. Intent.ACTION_MEDIA_UNMOUNTABLE;
  93. //
  94. Intent.ACTION_MEDIA_UNMOUNTED
  95. // 广播:扩展介质存在,但是还没有被挂载 (mount)。
  96. Intent.ACTION_NEW_OUTGOING_CALL;
  97.  
  98. Intent.ACTION_PACKAGE_ADDED;
  99. //成功的安装APK之后
  100. //广播:设备上新安装了一个应用程序包。
  101. //一个新应用包已经安装在设备上,数据包括包名(最新安装的包程序不能接收到这个广播)
  102. Intent.ACTION_PACKAGE_CHANGED;
  103. //一个已存在的应用程序包已经改变,包括包名
  104. Intent.ACTION_PACKAGE_DATA_CLEARED;
  105. //清除一个应用程序的数据时发出的广播(在设置--应用管理--选中某个应用,之后点清除数据时?)
  106. //用户已经清除一个包的数据,包括包名(清除包程序不能接收到这个广播)
  107.  
  108. Intent.ACTION_PACKAGE_INSTALL;
  109. //触发一个下载并且完成安装时发出的广播,比如在电子市场里下载应用?
  110. //
  111. Intent.ACTION_PACKAGE_REMOVED;
  112. //成功的删除某个APK之后发出的广播
  113. //一个已存在的应用程序包已经从设备上移除,包括包名(正在被安装的包程序不能接收到这个广播)
  114.  
  115. Intent.ACTION_PACKAGE_REPLACED;
  116. //替换一个现有的安装包时发出的广播(不管现在安装的APP比之前的新还是旧,都会发出此广播?)
  117. Intent.ACTION_PACKAGE_RESTARTED;
  118. //用户重新开始一个包,包的所有进程将被杀死,所有与其联系的运行时间状态应该被移除,包括包名(重新开始包程序不能接收到这个广播)
  119. Intent.ACTION_POWER_CONNECTED;
  120. //插上外部电源时发出的广播
  121. Intent.ACTION_POWER_DISCONNECTED;
  122. //已断开外部电源连接时发出的广播
  123. Intent.ACTION_PROVIDER_CHANGED;
  124. //
  125.  
  126. Intent.ACTION_REBOOT;
  127. //重启设备时的广播
  128.  
  129. Intent.ACTION_SCREEN_OFF;
  130. //屏幕被关闭之后的广播
  131.  
  132. Intent.ACTION_SCREEN_ON;
  133. //屏幕被打开之后的广播
  134.  
  135. Intent.ACTION_SHUTDOWN;
  136. //关闭系统时发出的广播
  137.  
  138. Intent.ACTION_TIMEZONE_CHANGED;
  139. //时区发生改变时发出的广播
  140.  
  141. Intent.ACTION_TIME_CHANGED;
  142. //时间被设置时发出的广播
  143.  
  144. Intent.ACTION_TIME_TICK;
  145. //广播:当前时间已经变化(正常的时间流逝)。
  146. //当前时间改变,每分钟都发送,不能通过组件声明来接收,只有通过Context.registerReceiver()方法来注册
  147.  
  148. Intent.ACTION_UID_REMOVED;
  149. //一个用户ID已经从系统中移除发出的广播
  150. //
  151.  
  152. Intent.ACTION_UMS_CONNECTED;
  153. //设备已进入USB大容量储存状态时发出的广播?
  154.  
  155. Intent.ACTION_UMS_DISCONNECTED;
  156. //设备已从USB大容量储存状态转为正常状态时发出的广播?
  157.  
  158. Intent.ACTION_USER_PRESENT;
  159. //
  160.  
  161. Intent.ACTION_WALLPAPER_CHANGED;
  162. //设备墙纸已改变时发出的广播

4.本节小结:

好的,关于BroadcastReceiver的学习就到这里,如果你有什么补充或者建议,欢迎提出~ 万分感激~
还没有评论.