8.广播

广播接收者

  • 现实中:电台要发布消息,通过广播把消息广播出去,使用收音机,就可以收听广播,得知这条消息
  • Android中:系统在运行过程中,会产生会多事件,那么某些事件产生时,比如:电量改变、收发短信、拨打电话、屏幕解锁、开机,系统会发送广播,只要应用程序接收到这条广播,就知道系统发生了相应的事件,从而执行相应的代码。使用广播接收者,就可以收听广播

创建广播接收者

  1. 定义java类继承BroadcastReceiver
  2. 在清单文件中定义receiver节点,定义name属性,指定广播接收者java类的全类名
  3. 在intent-filter的节点中,指定action子节点,action的值必须跟要接受的广播中的action匹配,比如,如果要接受打电话广播,
    那么action的值必须指定为
    
     <action android:name="android.intent.action.NEW_OUTGOING_CALL"/>
    
  • 因为打电话广播中所包含的action,就是"android.intent.action.NEW_OUTGOING_CALL",所以我们定义广播接收者时,
    action必须与其匹配,才能收到这条广播
  • 即便广播接收者所在进程已经被关闭,当系统发出的广播中的action跟该广播接收者的action匹配时,系统会启动该广播接收者所在的进程,
    并把广播发给该广播接收者

广播俩种注册方法

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

动态注册:需要使用广播接收者时,执行注册的代码,不需要时,执行解除注册的代码

  • 安卓中有一些广播接收者,必须使用代码注册,清单文件注册是无效的
  1. 屏幕锁屏和解锁
  2. 电量改变
  1. public class MainActivity extends Activity {
  2. private IntentFilter intentFilter;
  3. private NetworkChangeReceiver networkChangeReceiver;
  4. @Override
  5. protected void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.activity_main);
  8. intentFilter = new IntentFilter();
  9. intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
  10. networkChangeReceiver = new NetworkChangeReceiver();
  11. registerReceiver(networkChangeReceiver, intentFilter);
  12. }
  13. @Override
  14. protected void onDestroy() {
  15. super.onDestroy();
  16. unregisterReceiver(networkChangeReceiver);
  17. }
  18. class NetworkChangeReceiver extends BroadcastReceiver {
  19. @Override
  20. public void onReceive(Context context, Intent intent) {
  21. Toast.makeText(context, "network changes",Toast.LENGTH_SHORT).show();
  22. }
  23. }
  24. }

静态注册:下面全是

可以使用清单文件注册

  • 广播一旦发出,系统就会去所有清单文件中寻找,哪个广播接收者的action和广播的action是匹配的,如果找到了,就把该广播接收者的进程启动起来

案例1:IP拨号器

原理:接收拨打电话的广播,修改广播内携带的电话号码

  • 定义广播接收者接收打电话广播

public class CallReceiver extends BroadcastReceiver {

 

    //当广播接收者接收到广播时,此方法会调用

    @Override

    public void onReceive(Context context, Intent intent) {

        //拿到用户拨打的号码

        String number = getResultData();

        //修改广播内的号码

        setResultData("17951" + number);

    }

}
  • 在清单文件中定义该广播接收者接收的广播类型

    
      <receiver android:name="com.itheima.ipdialer.CallReceiver">
    
          <intent-filter >
    
              <action android:name="android.intent.action.NEW_OUTGOING_CALL"/>
    
          </intent-filter>
    
      </receiver>
    
  • 接收打电话广播需要权限
    
      <uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS"/>
    
  • 即使广播接收者的进程没有启动,当系统发送的广播可以被该接收者接收时,系统会自动启动该接收者所在的进程

案例2:短信拦截器

系统收到短信时会产生一条广播,广播中包含了短信的号码和内容

  • 系统发送短信广播时,是怎么把短信内容存入广播的,我们就只能怎么取出来
  • 如果短信过长,那么发送时会拆分成多条短信发送,那么短信广播中就会包含多条短信
  • 定义广播接收者接收短信广播

    
      public void onReceive(Context context, Intent intent) {
    
      //拿到广播里携带的短信内容
    
      Bundle bundle = intent.getExtras();
    
      Object[] objects = (Object[]) bundle.get("pdus");
    
      for(Object ob : objects ){
    
          //通过object对象创建一个短信对象
    
          SmsMessage sms = SmsMessage.createFromPdu((byte[])ob);
    
          System.out.println(sms.getMessageBody());
    
          System.out.println(sms.getOriginatingAddress());
    
      }
    

    }

  • 系统创建广播时,把短信存放到一个数组,然后把数据以pdus为key存入bundle,再把bundle存入intent
  • 清单文件中配置广播接收者接收的广播类型,注意要设置优先级属性,要保证优先级高于短信应用,才可以实现拦截
    
      <receiver android:name="com.itheima.smslistener.SmsReceiver">
    
          <intent-filter android:priority="1000">
    
              <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
    
          </intent-filter>
    
      </receiver>
    
  • 添加权限
    
      <uses-permission android:name="android.permission.RECEIVE_SMS"/>
    
  • 4.0以后广播接收者安装以后必须手动启动一次,否则不生效
  • 4.0以后广播接收者如果被手动关闭,就不会再启动了

项目3:监听SD卡状态

  • 清单文件中定义广播接收者接收的类型,监听SD卡常见的三种状态,所以广播接收者需要接收三种广播

    
       <receiver android:name="com.itheima.sdcradlistener.SDCardReceiver">
    
          <intent-filter >
    
              <action android:name="android.intent.action.MEDIA_MOUNTED"/>
    
              <action android:name="android.intent.action.MEDIA_UNMOUNTED"/>
    
              <action android:name="android.intent.action.MEDIA_REMOVED"/>
    
              <data android:scheme="file"/>
    
          </intent-filter>
    
      </receiver>
    
  • 广播接收者的定义
    
      public class SDCardReceiver extends BroadcastReceiver {
    
          @Override
    
          public void onReceive(Context context, Intent intent) {
    
              // 区分接收到的是哪个广播
    
              String action = intent.getAction();
    
     
    
              if(action.equals("android.intent.action.MEDIA_MOUNTED")){
    
                  System.out.println("sd卡就绪");
    
              }
    
              else if(action.equals("android.intent.action.MEDIA_UNMOUNTED")){
    
                  System.out.println("sd卡被移除");
    
              }
    
              else if(action.equals("android.intent.action.MEDIA_REMOVED")){
    
                  System.out.println("sd卡被拔出");
    
              }
    
          }
    
      }
    

项目4:勒索软件

  1. //在Activity中重写此方法,按返回键退不出去,但是可以菜单键退出,所以还需要开机自启
  2. @Override
  3. public void onBackPressed() {
  4. // super.onBackPressed();
  5. }
  • 接收开机广播,在广播接收者中启动勒索的Activity
  • 清单文件中配置接收开机广播
    
      <receiver android:name="com.itheima.lesuo.BootReceiver">
    
          <intent-filter >
    
              <action android:name="android.intent.action.BOOT_COMPLETED"/>
    
          </intent-filter>
    
      </receiver>
    
  • 权限
    
      <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
    
  • 定义广播接收者
    
      @Override
    
      public void onReceive(Context context, Intent intent) {
    
          //开机的时候就启动勒索软件
    
          Intent it = new Intent(context, MainActivity.class);        
    
          context.startActivity(it);
    
      }
    
  • 以上代码还不能启动MainActivity,因为广播接收者的启动,并不会创建任务栈,那么没有任务栈,就无法启动activity
  • 手动设置创建新任务栈的flag
    
      it.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    

项目5:监听应用的安装、卸载、更新

原理:应用在安装卸载更新时,系统会发送广播,广播里会携带应用的包名

  • 清单文件定义广播接收者接收的类型,因为要监听应用的三个动作,所以需要接收三种广播

    <receiver android:name="com.itheima.app.AppReceiver">

        <intent-filter >

            <action android:name="android.intent.action.PACKAGE_ADDED"/>

            <action android:name="android.intent.action.PACKAGE_REPLACED"/>

            <action android:name="android.intent.action.PACKAGE_REMOVED"/>

            <data android:scheme="package"/>

        </intent-filter>

    </receiver>
  • 广播接收者的定义

    
      public void onReceive(Context context, Intent intent) {
    
          //区分接收到的是哪种广播
    
          String action = intent.getAction();
    
          //获取广播中包含的应用包名
    
          Uri uri = intent.getData();
    
          if(action.equals("android.intent.action.PACKAGE_ADDED")){
    
              System.out.println(uri + "被安装了");
    
          }
    
          else if(action.equals("android.intent.action.PACKAGE_REPLACED")){
    
              System.out.println(uri + "被更新了");
    
          }
    
          else if(action.equals("android.intent.action.PACKAGE_REMOVED")){
    
              System.out.println(uri + "被卸载了");
    
          }
    
      }
    


自定义广播

发送自定义广播

  1. //发送自定义广播
  2. Intent intent = new Intent();
  3. //广播中的action也是自定义的
  4. intent.setAction("com.itheima.zdy");
  5. sendBroadcast(intent);

接收自定义广播

  1. //在清单文件注册,匹配自定义的广播,这个广播接收者就能接收到自定义的广播
  2. <receiver android:name="com.itheima.receivezdy.ZDYReceiver">
  3. <intent-filter >
  4. <action android:name="com.itheima.zdy"/>
  5. </intent-filter>

广播的分类

无序广播(标准广播)
  • 所有与广播中的action匹配的广播接收者都可以收到这条广播,并且是没有先后顺序,视为同时收到

有序广播

  • 所有与广播中的action匹配的广播接收者都可以收到这条广播,但是是有先后顺序的,按照广播接收者的优先级排序
    • 优先级的定义:-1000~1000
      1. <intent-filter android:priority="100" >
      2. <action android:name="com.example.broadcasttest.MY_BROADCAST"/>
      3. </intent-filter>
    • 最终接收者:所有广播接收者都接收到广播之后,它才接收,并且一定会接收
    • abortBroadCast:阻止其他接收者接收这条广播,类似拦截,只有有序广播可以被拦截

发送有序广播:

  1. sendOrderedBroadcast(intent);

第一个参数仍然是Intent,第二个参数是一个与权限相关的字符串,这里传入 null就行了

2. sendOrderedBroadcast(intent, null, new MyReceiver(), null, 0, "每人发100斤大米", null);

这里的第三个参数是最终接收者resultReceiver:不需要在清单文件中配置,这个广播接收者只接受该条有序广播,并且是最后一个收到该广播,并且一定可以收到该广播

  1. class MyReceiver extends BroadcastReceiver{
  2. @Override
  3. public void onReceive(Context context, Intent intent) {
  4. String text = getResultData();
  5. System.out.println("收到文件:" + text);
  6. }
  7. }
  1. //接收到自己发送出去的广播,最后接收(即使截断也能收到),因为在其他广播接收者可以修改数据
  2. public class ShengZF extends BroadcastReceiver {
  3. @Override
  4. public void onReceive(Context context, Intent intent) {
  5. // TODO Auto-generated method stub
  6. String text = getResultData();
  7. setResultData("每人发80斤大米");
  8. }
  9. }

用广播实现强制下线功能

实现强制下线功能

思路:需要在界面上弹出一个对话框,让用户无法进行任何其他操作,必须要点击对话框中的确定按钮,然后回到登录界面即可。

可是这样就存在着一个问题,因为被通知需要强制下线时可能正处于任何一个界面,难道需要在每个界面上都编写一个弹出对话框的逻辑?

不是的,我们可以借助广播知识,来实现这一功能。

1.强制下线功能需要先关闭掉所有的活动,然后回到登录界面。先创建一个ActivityCollector类用于管理所有的活动

2.创建BaseActivity类作为所有活动的父类

3.创建一个登录界面的布局login.xml

4.编写登录界面的活动,新建LoginActivity继承自BaseActivity

5.登录成功后进入程序主界面,这里不需要在主界面里提供什么功能,只需要加入强制下线功能就可以了,修改activity_main.xml中的代码,就是一个按钮

6.修改MainActivity中的代码按钮的点击事件里面发送了一条广播

7.创建一个广播接收器了,新建ForceOfflineReceiver

8.对AndroidManifest.xml文件进行配置:声明权限、对LoginActivity进行注册,并把它设置为主活动,最后再对 ForceOfflineReceiver 进行注册,

并指定它接收 com.example.broadcastbestpractice.FORCE_OFFLINE这条广播。  

  1. 这是第6步
  2. //在按钮的点击事件里面发送了一条广播,广播的值为com.example.broadcastbestpractice.FORCE_OFFLINE,这条广播就是用于通知程序强制用户下线的。
  3. //也就是说强制用户下线的逻辑并不是写在MainActivity里的,而是应该写在接收这条广播的广播接收器里面,这样强制下线的功能就不会依附于任何的界面,不管是在程序的任何地方,
  4. //只需要发出这样一条广播,就可以完成强制下线的操作了。
  5. public class MainActivity extends Activity
  6. {
  7. @Override
  8. public void onCreate(Bundle savedInstanceState)
  9. {
  10. super.onCreate(savedInstanceState);
  11. setContentView(R.layout.main);
  12. Button forceOffline = (Button) findViewById(R.id.force_offline);
  13. forceOffline.setOnClickListener(new OnClickListener() {
  14. @Override
  15. public void onClick(View v) {
  16. Intent intent = new Intent("com.example.broadcastbestpractice.FORCE_OFFLINE ");
  17. sendBroadcast(intent);
  18. }
  19. });
  20. }
  1. //第7步
  2. public class ForceOfflineReceiver extends BroadcastReceiver
  3. {
  4. @Override
  5. public void onReceive(final Context context, Intent intent) {
  6. AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(context);
  7. dialogBuilder.setTitle("Warning");
  8. dialogBuilder.setMessage("You are forced to be offline. Please tryto login again.");
  9. dialogBuilder.setCancelable(false);
  10. dialogBuilder.setPositiveButton("OK",new DialogInterface.OnClickListener() {
  11. @Override
  12. public void onClick(DialogInterface dialog, int which) {
  13. ActivityCollector.finishAll(); // 销毁所有活动
  14. Intent intent = new Intent(context,LoginActivity.class);
  15. intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  16. context.startActivity(intent); // 重新启动LoginActivity
  17. }
  18. });
  19. AlertDialog alertDialog = dialogBuilder.create();
  20. // 需要设置AlertDialog的类型,保证在广播接收器中可以正常弹出
  21. alertDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
  22. alertDialog.show();
  23. }
  24. }

onReceive()方法里加入了较多的代码,首先是使用AlertDialog.Builder来构建一个对话框,注意这里一定要调用setCancelable()方法将对话框设为不可取消,

否则用户按一下Back键就可以关闭对话框继续使用程序了。然后使用setPositiveButton()方法来给对话框注册确定按钮,当用户点击了确定按钮时,

就调用ActivityCollector的finishAll()方法来销毁掉所有活动,并重新启动LoginActivity这个活动。另外,由于在广播接收器里启动活动的,

因此一定要给Intent加入 FLAG_ACTIVITY_NEW_TASK这个标志。最后,还需要把对话框的类型设为TYPE_SYSTEM_ALERT,不然它将无法在广播接收器里弹出。

对AndroidManifest.xml文件进行配置,这里有几点内容需要注意,

1.由于在ForceOfflineReceiver里弹出了一个系统级别的对话框,因此必须要声明 android.permission.SYSTEM_ALERT_WINDOW权限。

2.对LoginActivity进行注册,并把它设置为主活动。

3.对 ForceOfflineReceiver 进行注册,并指定它接收 com.example.broadcastbestpractice.FORCE_OFFLINE这条广播。

来自为知笔记(Wiz)

时间: 2024-10-25 19:27:59

8.广播的相关文章

同一vlan不同子网广播

其一.问题 PC1网段为192.168.1.1/24.PC2网段为192.168.1.2/24 SW1的E0/1.E0/2划到vlan10中 两台PC能互通么? 答案:不可以互通.vlan只对本地有效 其二.问题 PC1网段为192.168.1.1/24.PC2网段为192.168.1.2/24 SW1的E0/1.E0/2划到vlan10中 SW2的E0/1.E0/2划到vlan20中 两台PC能互通么? 答案:可以互通.vlan只对本地有效.出本地再封装 其三.问题 PC1网段为192.168

BroadcastReceiver广播接收器基础

BroadcastReceiver 广播接收器 广播接收器,主要是用于app注册一些指定事件的广播接收器,系统在这些事件被触发的时候,会通知到注册的广播接收器,然后广播接收器在onReceive函数中对接收到的通知进行处理:广播接收器的两种注册方式:静态注册:<receiver android:name=".SMSBroadcastReceiver" > <intent-filter> <action android:name="android.

广播域与冲突域

局域网技术中,冲突域与广播域这两个词对网络性能影响很大,下面让我们看一看这两个词是如何影响网络性能的. 1.局域网技术之冲突域  假如你想将车驶出高速公路,但每做一次尝试,都有一辆车挡住你的去路.如果强行出来,就会发生碰撞,这与使用带冲突检测的载波侦听多路访问(CSMA/CD)协议的以太网上发生的情况很相似.电气与电子工程师协会(IEEE)将CSMA/CD以太网定义成802.3标准,如今,该标准的使用遍及整个网络界.开放系统互联模型(OSI)第二层的介质访问控制(MAC)子层,就是使用CSMA/

单播、多播(组播)和广播的区别

单播.多播和广播单播"(Unicast)."多播"(Multicast)和"广播"(Broadcast)这三个术语都是用来描述网络节点之间通讯方式的术语.那么这些术语究竟是什么意思?区别何在? 1.单播:网络节点之间的通信就好像是人们之间的对话一样.如果一个人对另外一个人说话,那么用网络技术的术语来描述就是"单播",此时信息的接收和传递只在两个节点之间进行.单播在网络中得到了广泛的应用,网络上绝大部分的数据都是以单播的形式传输的,只是一

转: 从现实生活中理解什么是广播机制

来自:http://blog.sina.com.cn/s/blog_714338950100p4km.html 一听到广播我们第一感觉就会联想到小时候村里面的广播,每逢村里有什么活动都是通过广播发送的.收听收音机也是一种广播,在收音机中有很多个广播电台,每个广播电台播放的内容都不相同.接收广播时广播(发送方)并不在意我们(接收方)接收到广播时如何处理.好比我们收听交通电台的广播,电台中告诉我们现在在交通状况如何,但它并不关心我们接收到广播时做如何做出处理,这不是广播应该关心的问题,OK,到这里我

Netty利用ChannelGroup广播消息

在Netty中提供了ChannelGroup接口,该接口继承Set接口,因此可以通过ChannelGroup可管理服务器端所有的连接的Channel,然后对所有的连接Channel广播消息. Server端: public class BroadCastServer { public static void run(int port) { EventLoopGroup boss = new NioEventLoopGroup(); EventLoopGroup worker = new NioE

手机影音第十六天,集成eventbus代替广播

代码已经托管到码云上,有兴趣的小伙伴可以下载看看 https://git.oschina.net/joy_yuan/MobilePlayer 一 EventBus 3.0   ---利用eventbus代替广播来获取音乐的数据. EventBus是一款针对Android优化的发布/订阅事件总线.主要功能是替代Intent,Handler,BroadCast在Fragment,Activity,Service,线程之间传递消息.优点是开销小,代码更优雅.以及将发送者和接收者解耦. 1.下载Even

Android广播机制(转)

1.Android广播机制概述 Android广播分为两个方面:广播发送者和广播接收者,通常情况下,BroadcastReceiver指的就是广播接收者(广播接收器).广播作为Android组件间的通信方式,可以使用的场景如下:1.同一app内部的同一组件内的消息通信(单个或多个线程之间): 2.同一app内部的不同组件之间的消息通信(单个进程): 3.同一app具有多个进程的不同组件之间的消息通信: 4.不同app之间的组件之间消息通信: 5.Android系统在特定情况下与App之间的消息通

洪泛和广播的区别

广播包的目的主机是全网用户,使用广播地址,在所有端口发送数据包,行为是主动的,可以理解为三层的行为: 泛洪是指交换机在MAC表中无法找到与数据包目标地址一致的条目,就将数据包从所有端口发送出去(除了接收该数据包的端口),以期找到目标主机来接收数据包,可以了解为2层的行为: 洪泛不要求维护网络的拓扑结构和相关的路由计算,仅要求接收到信息的节点以广播方式转发数据包.例如,源节点希望发送一段数据给目标节点.源节点首先通过网络将数据副本传送给它的每个邻居节点,每个邻居节点再将数据传送给各自的除发送数据来

冲突域 广播域

一.概念理解: 1.冲突域(物理分段): 连接在同一导线上的所有工作站的集合,或者说是同一物理网段上所有节点的集合或以太网上竞争同一带宽的节点集合.这个域代表了冲突在其中发生并传播的区域,这个区域可以被认为是共享段.在OSI模型中,冲突域被看作是第一层的概念,连接同一冲突域的设备有Hub,Reperter或者其他进行简单复制信号的设备.也就是说,用Hub或者Repeater连接的所有节点可以被认为是在同一个冲突域内,它不会划分冲突域.而第二层设备(网桥,交换机)第三层设备(路由器)都可以划分冲突