Android学习笔记(十三)

Android中的广播机制

Android提供了一套完整的API,允许应用程序自由地发送和接受广播。

发送广播的方法借助于Intent,接受广播的方法需要广播接收器(BroadcastsReceiver)。

Android中的广播主要分为两种类型,标准广播和有序广播。

 

标准广播(Normal broadcasts)是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在

同一时刻接受到这条广播消息。

有序广播(Ordered broadcasts)是一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能够收到这条广播的消息,

当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。(这样,前面的广播接收器就可以截断正在传递的广播)

 

广播接收器可以自由地对自己感兴趣的广播进行注册,这样当有相应的广播发出时,广播接收器就能够收到该广播,

并在内部处理相应的逻辑。

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

 

广播接收器的创建:新建一个类,让其继承自BroadcastsReceiver,并重写父类的onReceive()方法。

这样当有广播到来时,onReceive()方法就会执行,具体的逻辑就可以在这个方法中处理。

 

下面通过动态注册的方法编写一个能够监听网络变化的程序。新建一个BroadcastTest项目,

然后修改MainActivity中的代码,如下所示:

 1 public class MainActivity extends AppCompatActivity {
 2 
 3     private IntentFilter intentFilter;
 4 
 5     private NetworkChangeReceiver networkChangeReceiver;
 6 
 7     @Override
 8     protected void onCreate(Bundle savedInstanceState) {
 9         super.onCreate(savedInstanceState);
10         setContentView(R.layout.activity_main);
11         //创建一个IntentFilter的实例
12         intentFilter = new IntentFilter();
13         //广播接收器想要监听什么广播,就在这里添加相应的action
14         intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
15         //创建一个NetworkChangeReceiver的实例
16         networkChangeReceiver = new NetworkChangeReceiver();
17         //调用registerReceiver()方法进行注册,将NetworkChangeReceiver的实例和IntentFilter的实例都传递进去
18         registerReceiver(networkChangeReceiver,intentFilter);
19 
20     }
21 
22     @Override
23     public boolean onCreateOptionsMenu(Menu menu) {
24         // Inflate the menu; this adds items to the action bar if it is present.
25         getMenuInflater().inflate(R.menu.menu_main, menu);
26         return true;
27     }
28 
29     @Override
30     public boolean onOptionsItemSelected(MenuItem item) {
31         // Handle action bar item clicks here. The action bar will
32         // automatically handle clicks on the Home/Up button, so long
33         // as you specify a parent activity in AndroidManifest.xml.
34         int id = item.getItemId();
35 
36         //noinspection SimplifiableIfStatement
37         if (id == R.id.action_settings) {
38             return true;
39         }
40 
41         return super.onOptionsItemSelected(item);
42     }
43 
44     @Override
45     protected void onDestroy() {
46         super.onDestroy();
47         //动态注册的广播接收器一定要取消注册
48         unregisterReceiver(networkChangeReceiver);
49     }
50 
51     //定义一个内部类
52     class NetworkChangeReceiver extends BroadcastReceiver {
53         @Override
54         public void onReceive(Context context, Intent intent) {
55             //通过getSystemService()方法得到ConnectivityManager的实例,这是一个系统服务类,
56             // 专门用于管理网络连接的
57             ConnectivityManager connectivityManager =
58                     (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
59             //调用ConnectivityManager的getActiveNetworkInfo()方法可以得到NetworkInfo的实例
60             NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
61             //调用NetworkInfo的isAvailable()方法,就可以判断当前是否有网络了
62             if (networkInfo != null && networkInfo.isAvailable()) {
63                 Toast.makeText(context,"network is available",Toast.LENGTH_SHORT).show();
64             }else {
65                 Toast.makeText(context,"network is unavailable",Toast.LENGTH_SHORT).show();
66             }
67 
68         }
69     }
70 }

上面的代码中可以看到,在MainActivity中定义了一个内部类NetworkChangeReceiver,这个类是继承自BroadcastsReceiver的,

并重写了onReceive()方法。这样每当系统的网络状态发生变化时,onReceive()中的方法就会执行。

  在onReceive()方法中,首先通过getSystemService()方法得到ConnectivityManager的实例,这时一个系统服务类,专门用于

  管理网络的。

  然后调用ConnectivityManager的getActiveNetworkInfo()方法得到NetworkInfo的实例,接着调用NetworkInfo的isAvailable()方法,

  就可以判断是否有网络了。

onCreate()方法中,首先创建了一个IntentFilter的实例,给它添加一个值为android.net.conn.CONNECTIVITY_CHANGE的action。网络

发生变化时,系统发出的正是一条值为android.net.conn.CONNECTIVITY_CHANGE的广播。

接下来创建一个NetworkChangeReceiver的实例,然后调用registerReceiver()方法进行注册,将NetworkChangeReceiver的实例和

IntentFilter的实例都传递进去,这样NetworkChangeReceiver就会收到所有值为android.net.conn.CONNECTIVITY_CHANGE的广播。

 

注意:动态注册的广播接收器一定要取消注册才行,这里在onDestroy()方法中通过调用unregisterReceiver()方法来实现。

代码如下所示:

    protected void onDestroy() {super.onDestroy();//动态注册的广播接收器一定要取消注册
        unregisterReceiver(networkChangeReceiver);}

 

Android系统为了保证应用程序的安全性做了规定,如果程序要访问一些系统的关键性信息,必须要在AndroidManifest.xml文件中声明权限才可以。

例如下面的代码中就为查询系统的网络状态声明的权限,代码如下所示:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"package="com.mfeng.glh.broadcasttest" ><!-- 添加查询系统网络状态的权限声明 --><uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /><applicationandroid:allowBackup="true"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:theme="@style/AppTheme" ><activityandroid:name=".MainActivity"android:label="@string/app_name" ><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity></application></manifest>

运行程序,在系统设置中切换网络就可以收到广播消息了。

 

动态注册的广播可以自由的控制注册和注销,但是必须要程序启动以后才能接收广播。

如果想让程序在未启动的情况下接收广播,就要用的静态注册。

下面就使用静态注册的方法,接收一条开机广播,从而实现开机启动的功能。

新建一个BootCompleteReceiver类继承自BroadcastsReceiver,代码如下所示:

public class BootCompleteReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Toast.makeText(context,"Boot Complete",Toast.LENGTH_LONG).show();}
}

这里就不在使用内部类的方式来定义广播接收器了。在onReceive()方法中,简单的弹出一段提示信息。

然后修改AndroidManifest.xml文件,代码如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"package="com.mfeng.glh.broadcasttest" ><!-- 添加查询系统网络状态的权限声明 --><uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /><uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" /><applicationandroid:allowBackup="true"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:theme="@style/AppTheme" ><activityandroid:name=".MainActivity"android:label="@string/app_name" ><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity><receiver android:name=".BootCompleteReceiver" ><intent-filter><action android:name="android.intent.action.BOOT_COMPLETED" /></intent-filter></receiver></application></manifest>

上面的代码中,在<application>标签内出现了一个新的标签<receiver>,所有静态注册的广播接收器都是在这里进行注册的。

其中android:name指定具体注册哪一个广播接收器,然后在<intent-filter>标签中加入想要接收的广播就行了。

注意:监听系统开机广播也是需要声明权限的。

重新运行程序,就可以接收开机广播了。打开系统的应用程序管理界面,查看一下当前程序所拥有的权限,

如下图所示:

从图中可以看到,程序查看网络连接状态和开机启动的权限。

转载于:https://www.cnblogs.com/mffeng/p/4743706.html

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

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

相关文章

mysql offset函数_mysql查询语句解析

原标题&#xff1a;mysql查询语句解析一、查询基本知识select 列1,列2,...,列n from 表名1.条件查询&#xff1a;where①比较运算符&#xff1a;、!、< >、< 、>&#xff1b;②模糊查询&#xff1a;like/not like (%匹配任意多个字符,_匹配任意单个字符)&#xff1…

LeetCode 2013. 检测正方形(字典)

文章目录1. 题目2. 解题2.1 超时2.1 改进1. 题目 给你一个在 X-Y 平面上的点构成的数据流。设计一个满足下述要求的算法&#xff1a; 添加 一个在数据流中的新点到某个数据结构中。可以添加 重复 的点&#xff0c;并会视作不同的点进行处理。给你一个查询点&#xff0c;请你从…

ftp上传文件出现“sftp: cannot open /usr/SmartStore_test/1.zip to write”

转载链接:http://woshiniezhenhua.blog.163.com/blog/static/1913769282012103021831223/ 我使用的是win下的Xshell 4来进行个人台式机与机房分布式系统进行交互管理&#xff0c;主要涉及到上传下载文件&#xff0c;想上传代码时使用put命令上传文件&#xff0c;就是不行&#…

python 获取英文人名翻译

# 获取中英文人名翻译 import time import urllib.requestdef getename(ename_dataename2cname.txt):flag "jerry.asp?id" # 特定标记位置url https://name.supfree.net/tom.asp?idalphas [chr(x) for x in range(ord(a), ord(z) 1)]with open(ename_data, w,…

用python配置文件_使用。Python中的Py配置文件,python

python中使用.py配置文件一、格式&#xff1a;​ 创建一个config.py文件​ 在文件中加配置:DEBUGTruedm_connect {"dm_host":"127.0.0.1","dm_name":"dbname","dm_user":"dbuser","dm_passwd":"d…

浅析Java内存模型

概述 Java内存模型的主要目标是定义程序中各个变量的访问规则&#xff0c;即在虚拟机中将变量存储到内存和从内存中取出变量这样的底层细节。此处的变量是线程共享的&#xff0c;存在竞争问题的。 Java内存模型规定了所有的变量都存储在主内存&#xff0c;每条线程还有自己的工…

fastapi quickstart学习

文章目录1. 安装包2. 编写代码3. 终端运行4. 文档5. 增加数据learn from https://fastapi.tiangolo.com/zh/#typer-fastapi 1. 安装包 # pip install fastapi # pip install uvicorn[standard]2. 编写代码 main.py from typing import Optional # typing 模块用于类型检查…

单列变双列css_css 两列布局中单列定宽单列自适应布局的6种思路

前面的话说起自适应布局方式&#xff0c;单列定宽单列自适应布局是最基本的布局形式。本文将从float、inline-block、table、absolute、flex和grid这六种思路来详细说明如何巧妙地实现布局float【思路一】float说起两列布局&#xff0c;最常见的就是使用float来实现。float浮动…

Android中Activity启动模式

Android中Activity启动模式一共是四种&#xff0c;这里我写一下我对其的理解: 1.standard&#xff08;默认&#xff09; 2.singleTop&#xff0c;如果某个Activity设置了该属性&#xff0c;且当前Activity就是这个Activity&#xff0c;在界面中再次打开它的时候&#xff0c;不会…

fastapi 用户指南(路径参数、查询参数、请求体)

文章目录1. 第一步1.1 小结2. 路径参数2.1 顺序很重要2.2 预设值2.3 包含路径的路径参数3. 查询参数3.1 查询参数类型转换4. 请求体learn from https://fastapi.tiangolo.com/zh/tutorial/1. 第一步 pip install fastapi[all] from fastapi import FastAPI my_app FastAPI(…

mysql format row_MySQL之InnoDB存储引擎:Row Format行格式

MySQL下用的比较多、比较广的存储引擎就属InnoDB。这里我们来介绍下InnoDB存储引擎下数据记录的存储格式——Row Format行格式基本操作在MySQL中&#xff0c;所谓Row Format行格式是指数据记录(或者称之为行)在磁盘中的物理存储方式。具体地&#xff0c;对于InnoDB存储引擎而言…

DirectX API 编程起步 #01 项目设置

目录&#xff1a; DirectX API 编程起步 #02 窗口的诞生 DirectX API 编程起步 #02 创建3D世界 这里记录了我从零开始学习使用 DirectX API 的过程。 参考教程&#xff1a;rastertek -> DirectX 11 Tutorials 准备工作 1&#xff09;首先我们要有一套能写代码的软件&#…

centos5.8安装mysql5.3_CentOS 5.8 64位 源码安装mysql5.5.28

全新以最小化包安装了64位的CentOS5.8系统mysql5.5数据库v5.5.28地址&#xff1a;http://cdn.mysql.com/Downloads/MySQL-5.5/mysql-5.5.28.tar.gzmysql从5.5版本开始&#xff0c;不再使用./configure编译&#xff0c;而是使用cmake编译器我的mysql目录配置如下&#xff1a;安装…

LeetCode 2016. 增量元素之间的最大差值

文章目录1. 题目2. 解题1. 题目 给你一个下标从 0 开始的整数数组 nums &#xff0c;该数组的大小为 n &#xff0c;请你计算 nums[j] - nums[i] 能求得的 最大差值 &#xff0c;其中 0 < i < j < n 且 nums[i] < nums[j] 。 返回 最大差值 。如果不存在满足要求…

LeetCode 2017. 网格游戏(前缀和)

文章目录1. 题目2. 解题1. 题目 给你一个下标从 0 开始的二维数组 grid &#xff0c;数组大小为 2 x n &#xff0c;其中 grid[r][c] 表示矩阵中 (r, c) 位置上的点数。 现在有两个机器人正在矩阵上参与一场游戏。 两个机器人初始位置都是 (0, 0) &#xff0c;目标位置是 (1,…

mysql ngram_MySQL ngram全文解析器

本教程将向您展示如何使用MySQL ngram全文解析器来支持中文&#xff0c;日文&#xff0c;韩文等表意语言的全文搜索。MySQL ngram全文解析器简介MySQL内置的全文解析器使用空格确定单词的开始和结束。当涉及汉语&#xff0c;日语或韩语等表意语言语言时&#xff0c;这是一个限制…

手机号、姓名、邮箱等合法性验证方法

//邮箱 (BOOL) validateEmail:(NSString *)email{NSString *emailRegex "\\w[a-zA-Z_]?\\.[a-zA-Z]{2,6}";NSPredicate *emailTest [NSPredicate predicateWithFormat:"SELF MATCHES %", emailRegex];return [emailTest evaluateWithObject:email];}//…

zabbix设置mysql登陆免报警_zabbix3.0 监控mysql服务免用户名密码登录的有关问题故障处理详细过程_mysql...

zabbix3.0 监控mysql服务免用户名密码登录的问题故障处理详细过程1&#xff0c;My.cnf中用户名密码无效在azure云上面&#xff0c;使用Zabbix监控mysql中&#xff0c;发现在/usr/local/mysql/my.cnf里面设置的默认用户名密码无效&#xff0c;出不来数据&#xff0c;而且在zabbi…

LeetCode 2018. 判断单词是否能放入填字游戏内(模拟)

文章目录1. 题目2. 解题1. 题目 给你一个 m x n 的矩阵 board &#xff0c;它代表一个填字游戏 当前 的状态。 填字游戏格子中包含小写英文字母&#xff08;已填入的单词&#xff09;&#xff0c;表示 空格 的 和表示 障碍 格子的 # 。 如果满足以下条件&#xff0c;那么我…

链表的相关操作

#include<iostream> using namespace std; struct node {int data;node *next; }; //链表的建立&#xff0c;创建有n个结点的链表 node *create(int n) {node *headNULL;node *pNULL;headnew node();phead;cin>>p->data;node *q;while(--n){qnew node();cin>…