首页 > 移动平台 > 详细

安卓中广播介绍,广播的使用

时间:2015-10-14 18:14:53      阅读:280      评论:0      收藏:0      [点我收藏+]

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

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

标准广播( Normal broadcasts) 是一种完全异步执行的广播,在广播发出之后,所有的
广播接收器几乎都会在同一时刻接收到这条广播消息,因此它们之间没有任何先后顺序可
言。这种广播的效率会比较高,但同时也意味着它是无法被截断的。

 

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

 

1,动态注册监听网络变化
 那么该如何创建一个广播接收器呢?其实只需要新建一个类,让它继承自 BroadcastReceiver,
并重写父类的 onReceive()方法就行了。这样当有广播到来时, onReceive()方法就会得到执行,
具体的逻辑就可以在这个方法中处理。

public class MainActivity extends Activity {
 private IntentFilter intentFilter;
 private NetworkChangeReceiver networkChangeReceiver;
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  intentFilter = new IntentFilter(); 
// 因为当网络状态发生变化时,系统发出的正是一条值为 android.net.conn.CONNECTIVITY_
   CHANGE 的广播,   
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
  networkChangeReceiver = new NetworkChangeReceiver();
  registerReceiver(networkChangeReceiver, intentFilter);
 }
 @Override
 public boolean onCreateOptionsMenu(Menu menu) {
  // Inflate the menu; this adds items to the action bar if it is present.
  getMenuInflater().inflate(R.menu.main, menu);
  return true;
 }
 class NetworkChangeReceiver extends BroadcastReceiver {
  @Override
  public void onReceive(Context context, Intent intent) {
  Toast.makeText(context, "network changes",
  Toast.LENGTH_SHORT).show();
  }
 }
 @Override
 protected void onDestroy() {
  super.onDestroy();
  unregisterReceiver(networkChangeReceiver);
 }
}

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

 

2,动态注册的广播接收器可以自由地控制注册与注销,在灵活性方面有很大的优势,但是
它也存在着一个缺点,即必须要在程序启动之后才能接收到广播,因为注册的逻辑是写在
onCreate()方法中的。那么有没有什么办法可以让程序在未启动的情况下就能接收到广播
呢?这就需要使用静态注册的方式了。

这里我们准备让程序接收一条开机广播,当收到这条广播时就可以在 onReceive()方法里
执行相应的逻辑,从而实现开机启动的功能。新建一个 BootCompleteReceiver 继承自
BroadcastReceiver,代码如下所示:

public class BootCompleteReceiver extends BroadcastReceiver {

@Override

public void onReceive(Context context, Intent intent) {

Toast.makeText(context, "Boot Complete", Toast.LENGTH_LONG).show();

        }

    }

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

<manifest xmlns:android="http://schemas.android.com/apk/res/android"

package="com.example.broadcasttest"

android:versionCode="1"

android:versionName="1.0" >

……

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

//声明权限 
            <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

<application

android:allowBackup="true"

android:icon="@drawable/ic_launcher"

android:label="@string/app_name"

android:theme="@style/AppTheme" >

……

                <receiver android:name=".BootCompleteReceiver" >

                <intent-filter>

                //注册开机自动启动监听 
                <action android:name="android.intent.action.BOOT_COMPLETED" />

                </intent-filter>

                </receiver>

</application>

</manifest>

 

5.3 发送自定义广播
        5.3.1 发送标准广播

1,需要先定义一个广播接收器来准备接收此广播

public class MyBroadcastReceiver extends BroadcastReceiver {

    @Override

    public void onReceive(Context context, Intent intent) {

        Toast.makeText(context, "received in MyBroadcastReceiver",

            Toast.LENGTH_SHORT).show();

        }

    }

2,AndroidManifest.xml 中对这个广播接收器进行注册

<manifest xmlns:android="http://schemas.android.com/apk/res/android"

package="com.example.broadcasttest"

android:versionCode="1"

android:versionName="1.0" >

……

<application

android:allowBackup="true"

android:icon="@drawable/ic_launcher"

android:label="@string/app_name"

android:theme="@style/AppTheme" >

……

           
 <receiver android:name=".MyBroadcastReceiver">

            <intent-filter>

            <action android:name="com.example.broadcasttest. MY_BROADCAST"/>

            </intent-filter>

            </receiver>

</application>

</manifest>

3,布局文件中定义了一个按钮,用于作为发送广播的触发点。然后修改 MainActivity中的代码

Button button = (Button) findViewById(R.id.button);

button.setOnClickListener(new OnClickListener() {

        @Override

        public void onClick(View v) {

            Intent intent = new Intent("com.example.broadcasttest.

            MY_BROADCAST");

            sendBroadcast(intent);

}

5.3.1 发送有序广播

sendOrderedBroadcast(intent, null);
 
<receiver android:name=".MyBroadcastReceiver">
<intent-filter android:priority="100" >
<action android:name="com.example.broadcasttest.MY_BROADCAST"/>
</intent-filter>
</receiver>

我们通过 android:priority 属性给广播接收器设置了优先级,优先级比较高的
广播接收器就可以先收到广播。这里将 MyBroadcastReceiver 的优先级设成了 100,以保证它
一定会在 AnotherBroadcastReceiver 之前收到广播。

abortBroadcast();//阻止广播继续传递。

已经获得了接收广播的优先权,那么 MyBroadcastReceiver 就可以选择是否允许广
播继续传递了。

 

5.4 使用本地广播

public class MainActivity extends Activity {

private IntentFilter intentFilter;

private LocalReceiver localReceiver;

private LocalBroadcastManager localBroadcastManager;

    @Override

    protected void onCreate(Bundle savedInstanceState) {

    super.onCreate(savedInstanceState);

    setContentView(R.layout.activity_main);

    
localBroadcastManager = LocalBroadcastManager.getInstance(this);

    // 
获取实例

Button button = (Button) findViewById(R.id.button);

button.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

    Intent intent = new Intent("com.example.broadcasttest.

    LOCAL_BROADCAST");

    localBroadcastManager.sendBroadcast(intent); // 
发送本地广播

}

});

intentFilter = new IntentFilter();

        intentFilter.addAction("com.example.broadcasttest.LOCAL_BROADCAST");

        localReceiver = new LocalReceiver();

        localBroadcastManager.registerReceiver(localReceiver, intentFilter);

        // 
注册本地广播监听器

}

@Override

protected void onDestroy() {

super.onDestroy();

        localBroadcastManager.unregisterReceiver(localReceiver);

}

class LocalReceiver extends BroadcastReceiver {

@Override

public void onReceive(Context context, Intent intent) {

Toast.makeText(context, "received local broadcast",

Toast.LENGTH_SHORT).show();

        }

    }

}

本地广播是无法通过静态注册的方式来接收的。其实这也完全
可以理解,因为静态注册主要就是为了让程序在未启动的情况下也能收到广播,而发送本地
广播时,我们的程序肯定是已经启动了,因此也完全不需要使用静态注册的功能。

广播的应用实例:实现强制下线功能
其实实现强制下线功能的思路也比较简单,只需要
在界面上弹出一个对话框,让用户无法进行任何其他操作,必须要点击对话框中的确定按钮,
然后回到登录界面即可。可是这样就存在着一个问题,因为我们被通知需要强制下线时可能
正处于任何一个界面,难道需要在每个界面上都编写一个弹出对话框的逻辑?如果你真的这
么想,那思维就偏远了,我们完全可以借助本章中所学的广播知识,来非常轻松地实现这一
功能。新建一个 Broadcast







 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

安卓中广播介绍,广播的使用

原文:http://my.oschina.net/u/2480757/blog/517164

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!