Android入门(11)| 全局广播与本地广播

文章目录

  • 广播概念
  • 接收广播
    • 动态注册
      • 实例
    • 静态注册
      • 实例
  • 发送广播
    • 发送标准广播
    • 广播的跨进程特性
    • 发送有序广播
  • 本地广播


广播概念

Android 中的每个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收到自己所关心的广播内容,这些广播可能是来自系统的,也可能是来自于其他应用程序的。

广播有两种类型——有序广播和标准广播:

  • 标准广播: 一种完全异步执行的广播,在广播发出去之后,所有的广播接收器几乎都会同一时刻接收到这条广播消息,没有任何的先后顺序可言,这种广播的效率比较高,但是无法被截断

在这里插入图片描述

  • 有序广播: 是一种同步执行的广播,在广播发出去之后,同一时刻只会有一个广播接收器能够收到这条消息,当这个广播接收器中的逻辑执行完毕之后,广播才会继续传递,所以这时候的广播接收器是有优先级顺序的,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播就无法收到广播消息。

在这里插入图片描述

接收广播

动态注册

Android内置了很多系统级别的广播,我们可以在应用程序中通过监听这些广播来得到各种系统的状态信息。比如手机开机、电池电量发生变化、时间或者时区发生改变等等。如果想要接收到这些广播就需要使用广播接收器。

注册广播的方式一般也有两种,在 代码中注册(动态注册) 或者 AndroidManifest.xml 中注册(静态注册)

实例

我们实现一个能准确地告诉用户当前有没有网络的功能,在实现代码前,由于 Android 为了保护用户设备的隐私和安全,规定了程序需要进行一些对用户来说比较敏感的操作,必须在配置文件中声明权限才可以,该功能监听了网络的变化,所以必须在 AndroidManifest.xml 配置权限才能访问系统网络状态:
在这里插入图片描述

public class BroadcastActivity extends AppCompatActivity {private IntentFilter intentFilter; // 意图过滤器,旨在匹配可以响应对应操作的组件private NetworkChangeReceiver networkChangeReceiver; // 自定义的广播接收器类@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);intentFilter = new IntentFilter();// android.net.conn.CONNECTIVITY_CHANGE是网络状态变化时系统的广播intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");networkChangeReceiver = new NetworkChangeReceiver();// 注册,将接收器与IntentFilter相匹配registerReceiver(networkChangeReceiver, intentFilter);}// 动态注册一定要在结束时取消注册protected void onDestroy(){super.onDestroy();// 实现取消注册unregisterReceiver(networkChangeReceiver);}// 自定义的广播接收器类class NetworkChangeReceiver extends BroadcastReceiver{@Overridepublic void onReceive(Context context, Intent intent) {// 通过getSystemService得到管理网络连接的connectivityManager实例ConnectivityManager connectivityManager = (ConnectivityManager)getSystemService(Context.CONNECTIVITY_SERVICE);NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();// 通过isAvailable方法判断是否有网络if(networkInfo != null && networkInfo.isAvailable()){Toast.makeText(context, "network is available", Toast.LENGTH_SHORT).show();} else{Toast.makeText(context, "network is unavailable", Toast.LENGTH_SHORT).show();}}}
}

静态注册

动态注册的广播可以自由控制注册、注销,很灵活,缺点是必须在程序启动之后才能接受到广播,因为注册的逻辑是写在 onCreate() 里面的,使用静态注册可以让程序在未启动的情况下接受到广播

实例

静态注册需要在 AndroidManifest.xml 进行注册,使用 receiver 标签,并告诉这个 receiver 注册哪一个 action,下面是一个开机启动接受广播的案例:

在这里插入图片描述

public class BootCompleteReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// TODO: This method is called when the BroadcastReceiver is receiving// an Intent broadcast.Toast.makeText(context, "Boot Complete", Toast.LENGTH_SHORT).show();}
}

PS:不要在 onReceive() 方法中添加过多逻辑或耗时操作,因为广播接收器中不允许开启线程,因此当 onReceive() 运行较长时间却未结束时,程序就会报错。


发送广播

发送标准广播

发送广播使用 Intent 进行发送,首先需要准备一个接收器用于接受发送的广播:

public class MyBroadcastReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// 收到自定义广播时会弹出提示Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_LONG).show();}
}

AndroidManifest.xml 中注册广播的值:

实现点击 BroadcastActivity 活动中的 send broadcast 按钮来发送广播:

布局文件 broad_layout.xml

<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"android:orientation="vertical"android:layout_width="match_parent"android:layout_height="match_parent"><Buttonandroid:id="@+id/button_broadcast1"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="send broadcast"/>
</LinearLayout>

活动代码 BroadcastActivity.java

public class BroadcastActivity extends AppCompatActivity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.broad_layout);Button button = findViewById(R.id.button_broadcast1);button.setOnClickListener((View v)->{// 将要发送的广播植入IntentIntent intent = new Intent("com.example.activitytest.Activity.MY_BROADCAST");// 参数1:包名;参数2:接收器的路径ComponentName componentName = new ComponentName("com.example.activitytest","com.example.activitytest.Activity.MyBroadcastReceiver");// 通过调用Intent中的setComponent方法,我们可以打开另外一个应用中的Activity或者服务。intent.setComponent(componentName);// 调用Context的sendBroadcast方法发送广播sendBroadcast(intent);});}
}

PS: ComponentName 构造函数的第一个参数指的包名是 AndroidManifest.xml 文件下 package 属性对应的包名:
在这里插入图片描述
而非 BroadcastActivity.java 文件所在的包名:
在这里插入图片描述
详见该博客

运行结果:
在这里插入图片描述


广播的跨进程特性

新建一个项目,创建广播接收器 MyReceiver.java

public class MyReceiver extends BroadcastReceiver {private static final String TAG = "MyReceiver";@Overridepublic void onReceive(Context context, Intent intent) {Log.e(TAG, "onReceive: gone");Toast.makeText(context, "received gone", Toast.LENGTH_LONG).show();}
}

AndroidManifest.xml
在这里插入图片描述
在原来项目的 BroadcastActivity.java 文件中发送第二条广播:
在这里插入图片描述
运行结果:
在这里插入图片描述
在这里插入图片描述
更多关于广播的问题详见本文——解决 Android 8.0 以上静态广播无法注册


发送有序广播

很多人对之前的代码可能会有疑问,我指定广播发送给哪个包的哪个接收器,这还算“广播”吗?因此,对于安卓高版本而言,还有另一种发送广播的方法:

修改 BroadcastActivity.java 中的代码:
在这里插入图片描述

即可实现真正意义上的广播。在此基础上,我们发送有序广播,定义接收器的优先级
在这里插入图片描述
并在接收器 MyBroadcastReceiver.java 中截断广播,不允许广播继续传递:

public class MyBroadcastReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {// 收到自定义广播时会弹出提示Toast.makeText(context, "received in MyBroadcastReceiver", Toast.LENGTH_SHORT).show();// 截断abortBroadcast();}
}

不必对另一个接收器 MyReceiver 进行更改,此时就已达到了只有 MyBroadcastReceiver 能收到广播,而 MyReceiver 不能收到广播的目的。


本地广播

前面我们发送和接受的广播都是系统的全局广播,发出的广播可以被其他任何应用程序接收到。这样容易引起安全问题,为了解决安全性问题,Android 支持发送本地广播,其有以下特点:

  • 广播不会离开我们的程序,无需担心机密数据泄露;
  • 其他程序的广播无法发送到我们程序内部,无需担心有安全漏洞的隐患;
  • 比发送全局广播更高效。
  • 本地广播的接收只能使用动态注册,因为静态注册就是为了让程序在未启动的时候也能接收到广播,而发送本地广播的时候应用程序肯定启动了

本地广播并不复杂,主要就是使用了一个 LocalBroadcastManager 来对广播进行管理,并且提供了发送广播和注册广播接收器的方法:

public class BroadcastActivity extends AppCompatActivity {private static final String TAG = "BroadcastActivity";private IntentFilter intentFilter; // 意图过滤器private LocalReceiver localReceiver; // 自定义接收器类private LocalBroadcastManager localBroadcastManager; // support包提供的本地广播工具@SuppressLint("WrongConstant")@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.broad_layout);localBroadcastManager = LocalBroadcastManager.getInstance(this); // 获取实例Button button = findViewById(R.id.button_broadcast1);button.setOnClickListener((View v)->{Log.e(TAG, "onCreate: start");// 将要发送的广播植入IntentIntent intent = new Intent("com.example.activitytest.Activity.LOCAL_BROADCAST");if(Build.VERSION.SDK_INT >= 28){// 突破隐式广播限制intent.addFlags(0x01000000);}localBroadcastManager.sendBroadcast(intent); // 发送本地广播});// 动态注册的步骤intentFilter = new IntentFilter();// 添加自定义广播intentFilter.addAction("com.example.activitytest.Activity.LOCAL_BROADCAST");// 实例化接收器localReceiver = new LocalReceiver();// 注册接收器localBroadcastManager.registerReceiver(localReceiver, intentFilter);}// 动态注册一定要在结束时取消注册@Overrideprotected void onDestroy() {super.onDestroy();localBroadcastManager.unregisterReceiver(localReceiver);}class  LocalReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Toast.makeText(context, "本地广播", Toast.LENGTH_LONG).show();}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/443702.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android开发(2) | 广播 Broadcast 的应用——强制下线功能

文章目录功能简介关闭所有活动登陆界面发送强制下线的广播广播接收器AndroidManifest.xml运行结果功能简介 强制下线功能只需要弹出一个对话框&#xff0c;让用户只能点击确定按钮&#xff0c;回到登录界面。 如果在每一个活动中添加一个对话框的话太过繁琐&#xff0c;用广播…

Android入门(12)| 数据持久化

文章目录数据持久化文件存储将数据存储进文件实例从文件中读取数据实例SharedPreferences存储将数据存储进文件实例从文件中读取数据实例实现记住密码的功能SQLite数据库存储创建自己的帮助类调用自己的帮助类补全 onUpgrade() 方法增删查改增&#xff1a;SQLiteDatabase.inser…

Android入门(13)| Android权限 与 内容提供器

文章目录普通权限与危险权限运行时申请权限内容提供器运用安卓封装好的内容提供器自实现的内容提供器概念实现普通权限与危险权限 主要用于不同应用程序之间在保证被访数据的安全性的基础上&#xff0c;实现数据共享的功能。 在 Android 6.0 开始引入了运行时权限的功能&…

Android入门(14)| 通知

文章目录创建通知点击效果其它小功能实例创建通知 创建通知的步骤&#xff1a; 管理通知的 NotificationManager&#xff0c;通常通过当前 Context 的 getSystemService() 获取实例。它接受一个字符串参数用于确定获取系统的什么服务。Android 8.0(O) 版本后需要通知通道&…

Android开发(3) | 权限和内容提供器的应用——调用相机和相册

文章目录拍照并保存到 ImageView 控件布局文件 notice_layout.xml按钮 button_takePhoto 的点击操作隐式 Intent 启动后的回调AndroidManifest.xml从相册选取照片并在 ImageView 控件中显示布局文件 notice_layout.xml按钮 button_takePhoto 的点击操作自定义打开相册的方法 op…

Android开发(4) | 系统权限、MediaPlayer类 和 VideoView类 的应用——播放多媒体文件

文章目录MediaPlayer类播放音频的实例VideoView类播放视频的实例MediaPlayer类 对多种格式的音频文件提供了全面的控制方法&#xff1a; 如何获得MediaPlayer实例&#xff1f; 通过构造函数&#xff1a; MediaPlayer mp new MediaPlayer();调用 MediaPlayer.create() 方法&…

Android入门(15)| 网络

文章目录WebViewHTTP使用HttpURLConnection使用OkHttp封装网络操作封装HttpURLConnection封装OkHttpWebView WebView 可以在 应用程序中&#xff08;而不是浏览器&#xff09; 展示一些网页。 布局文件 web_layout.xml&#xff1a; <LinearLayoutxmlns:android"http…

Android入门(16)| 服务

文章目录概念Android 多线程继承 Thread继承 Runable 接口匿名类异步消息处理AsyncTask使用服务框架启动/停止服务绑定/解绑服务服务的生命周期前台服务IntentService完整版下载示例下载过程的回调接口&#xff1a;DownloadListener继承 AsyncTask 实现下载功能&#xff1a;Dow…

2020德勤面试开始了吗_2020国考面试开始,近期面试公告汇总,附结构化小组面试流程...

2020年国家公务员考试面试环节逐步恢复考试&#xff0c;各个招录部门已经发布面试考察公告&#xff0c;对于进入面试环节的国考考生来说&#xff0c;有必要了解近期国考面试的招录动态&#xff0c;提前做好面试准备。2020国考国家统计局机关面试面试确认&#xff1a;请进入面试…

项目积压需求项目计划_需求变更频繁,项目经理如何做好需求管理?

项目实施过程中&#xff0c;项目经理常常面临一个重大挑战——需求变更。需求变更无处不在&#xff0c;市场条件变化、新业务出现、战略目标调整、客户需求修改、资源限制等&#xff0c;都会造成需求变更。需求变更会影响项目的时间、成本和质量&#xff0c;对整个项目和团队成…

Android | Sensor.TYPE_ORIENTATION被废弃后的解决办法

文章目录概述getOrientation 方法根据 旋转矩阵R 获取 设备旋转弧度getRotationMatrix 方法根据 地磁场、加速度传感器对象 获取 旋转矩阵R代码参考资料概述 Sensor.TYPE_ORIENTATION 常数在 API 8 中已弃用&#xff0c;官方推荐使用 SensorManager.getOrientation() 替代。关…

【JAVA 开发小问题】 | String操作合集

文章目录截取特定两个字符之间的字符串截取特定两个字符之间的字符串 利用正则表达式&#xff0c;图片来源

uniapp 刷新后数据都没有了_环境温度传感器都没有连接,竟然还会有数据?

福田欧曼GTL(福康发动机、康明斯2880系统)匹配ECoffit尿素泵●故障现象&#xff1a;OBD故障灯点亮&#xff0c;不烧尿素&#xff0c;油耗高&#xff0c;动力不足●故障码&#xff1a;●维修分析&#xff1a;①故障指出加热器问题&#xff0c;摸下尿素箱温度&#xff0c;发现烫手…

保姆级教学!Xcode 配置 OpenGL 环境

文章目录GLFW获取 GLFWGLAD获取 GLAD在 Xcode 中配置下载好的 GLFW 和 GLAD配置流程检测是否配置成功无关配置的题外话——Xcode 下安全的删除移动操作GLFW Graphics Library Framework&#xff08;图形库框架&#xff09;&#xff0c;可以让我们通过其封装好的 通用API 来正确…

Android入门(17)| 百度提供的 Android定位SDK

文章目录配置百度提供的 Android定位SDK用于发布的 SHA1用于测试的 SHA1使用百度定位实例配置百度提供的 Android定位SDK 详情参见官方文档&#xff0c;这里仅对获取 SHA1 做详细介绍&#xff1a; 用于发布的 SHA1 用于测试的 SHA1 使用百度定位实例 public class LocationAc…

ios 不被遮挡 阴影_为何你没见到日环食?你不知道的天象常识原来还有这么多 | 返朴...

关注风云之声提升思维层次导读说好的日环食呢&#xff0c;为什么上周很多人只等到了日偏食?日食月食的时间和种类是怎么预测的?你真的弄懂了各种日食和月食的成因吗&#xff1f;你了解它们有什么区别和联系&#xff0c;又遵循什么样的时间规律吗? 日食和月食发生的频率一样吗…

初识贝塞尔(bezier)曲线

文章目录资料援引贝塞尔曲线的用途一阶贝塞尔&#xff08;bezier&#xff09;曲线二阶贝塞尔&#xff08;bezier&#xff09;曲线三阶贝塞尔&#xff08;bezier&#xff09;曲线高阶贝塞尔&#xff08;bezier&#xff09;曲线三阶贝塞尔曲线求插值&#xff08;Slerp&#xff09…

python代码测试 vim_用 Hypothesis 快速测试你的 Python 代码

点击上方“Python编程时光”&#xff0c;选择“加为星标”第一时间关注Python技术干货&#xff01;介绍无论你使用哪种编程语言或框架&#xff0c;测试都非常重要。Hypothesis是 Python 的一个高级测试库。它允许编写测试用例时参数化&#xff0c;然后生成使测试失败的简单易懂…

Mac 下 CMake 的配置与使用

文章目录安装与配置编译单个源文件编译前的准备开始编译编译多个源文件多个源文件在同一目录下多个源文件在不同目录下math 目录下的 CMakeLists.txt根目录的 CMakeLists.txtoption 选项导入外部库本地导入&#xff08;find_package&#xff09;外部导入&#xff08;FetchConte…

五轴编程_沙井万丰数控数控编程五轴编程那个软件好用

沙井万丰数控数控编程五轴编程那个软件好用设计需要掌握很高很全面的知识和技能&#xff0c;模具做的好&#xff0c;产品质量好&#xff0c;模具结构合理&#xff0c;生产效率高&#xff0c;工厂效益好。正因如此&#xff0c;模具技术工在外打工的工资都非常的高。少则每月几千…