二、广播事件监听的两种方法
使用广播进行事件监听有两种方法,静态注册和动态注册,又或者称冷插拔和热插拔。静态注册就是将广播接收器的相关信息写在应用的配置文件中。当有广播事件发生时,组件管理服务就会从安装包管理服务中获取已安装应用的广播组件信息。动态注册则是通过Context.registerReceiver和Context.unregisterRecever,动态将广播接收器与所需要监听的事件绑定。
<receiver android:name=".ColdReceiver"><!-- 你的Receiver名称 --> <intent-filter> <action android:name="android.intent.action.COLD_BROADCAST"/> <!-- 你广播要接受的intent名称 --> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </receiver>
然后我们建一个ColdReceiver的类,继承BroadcastReceiver,里面代码如下
public class ColdReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { //跳转到service中 intent = new Intent("android.intent.action.BroadcastService"); intent.addFlags(1); //开启service context.startService(intent); //日志打印 Log.d("TEST","静态注册"); } }
上面的Service启动看到了吗,这里我用了上次说过的快捷跳转。其中service的配置和内容如下
<service android:name=".BroadcastService"><!-- 你自定义的service文件 (在<application></application>里面加)--> <intent-filter> <action android:name="android.intent.action.BroadcastService" /><!-- 用intent启动时的快捷名(也可以用常规的方式启动) --> <category android:name="android.intent.category.default" /> </intent-filter> </service>
1 public class BroadcastService extends Service{ 2 3 @Override 4 public IBinder onBind(Intent intent) { 5 // TODO Auto-generated method stub 6 return null; 7 } 8 9 @Override 10 public void onCreate() { 11 //开启服务时会首先调用该方法 12 super.onCreate(); 13 } 14 15 @Override 16 public int onStartCommand(Intent intent, int flags, int startId) { 17 //根据每次intent传过来的信息进行判断来显示不同信息 18 switch(intent.getFlags()){ 19 case 1:{ 20 Toast.makeText(getApplicationContext(), "静态注册", Toast.LENGTH_SHORT).show(); 21 break; 22 } 23 case 2:{ 24 Toast.makeText(getApplicationContext(), "动态注册", Toast.LENGTH_SHORT).show(); 25 break; 26 } 27 case 3:{ 28 Toast.makeText(getApplicationContext(), "普通广播", Toast.LENGTH_SHORT).show(); 29 break; 30 } 31 case 4:{ 32 Toast.makeText(getApplicationContext(), "有序广播", Toast.LENGTH_SHORT).show(); 33 break; 34 } 35 } 36 return START_STICKY; 37 } 38 39 @Override 40 public void onDestroy() { 41 // 停止service后会调用此方法 42 Log.d("TEST", "destroy"); 43 super.onDestroy(); 44 } 45 46 }
那么静态广播的创建就完成了,简单吧,就两个步骤,一是配置广播,二是继承BroadcastReceiver,重写里面的onReceive函数。
接下来我们在新建一个工程来检测广播是否可以响应消息。(检测应用的代码全部都在下方,并且不用再做更改了)
1 public class MainActivity extends Activity implements OnClickListener{ 2 3 private Button b1,b2,b3,b4; 4 private Intent intent; 5 @Override 6 protected void onCreate(Bundle savedInstanceState) { 7 super.onCreate(savedInstanceState); 8 setContentView(R.layout.activity_main); 9 intent = new Intent(); 10 //获得界面的控件 11 b1 = (Button) findViewById(R.id.button1); 12 b1.setOnClickListener(this); 13 b2 = (Button) findViewById(R.id.button2); 14 b2.setOnClickListener(this); 15 b3 = (Button) findViewById(R.id.button3); 16 b3.setOnClickListener(this); 17 b4 = (Button) findViewById(R.id.button4); 18 b4.setOnClickListener(this); 19 Log.d("TEST","===初始化完成==="); 20 } 21 22 @Override 23 public void onClick(View v) { 24 // TODO Auto-generated method stub 25 switch(v.getId()){ 26 case R.id.button1:{//发送到静态注册广播 27 intent = new Intent("android.intent.action.COLD_BROADCAST"); 28 sendBroadcast(intent); 29 //intent.putExtra("msg", "hello coldreceiver."); 30 break; 31 } 32 case R.id.button2:{//发送到动态注册广播 33 intent = new Intent("android.intent.action.HOT_BROADCAST"); 34 //intent.putExtra("msg", "hello hotreceiver."); 35 sendBroadcast(intent); 36 break; 37 } 38 case R.id.button3:{//普通广播 39 intent = new Intent("android.intent.action.NORMAL_BROADCAST"); 40 sendBroadcast(intent); 41 break; 42 } 43 case R.id.button4:{//有序广播 44 intent = new Intent("android.intent.action.SORT_BROADCAST"); 45 sendOrderedBroadcast(intent, "scott.permission.SORT_BROADCAST_PERMISSION"); 46 break; 47 } 48 } 49 } 50 51 public void show(String str){ 52 Toast.makeText(this, str, Toast.LENGTH_LONG).show(); 53 } 54 55 @Override 56 protected void onDestroy() { 57 // TODO Auto-generated method stub 58 super.onDestroy(); 59 } 60 }
ok,将两个应用都安装到设备上,启动测试用的应用,点击第一个按钮,运行的效果如下
同时,会出现"静态注册"的Toast(不方便截图)。可以看出静态注册广播能够跨应用来响应信息,这都要归功于安卓上的组件管理服务,它会读取每个应用的配置文件,然后获取里面的组件信息,每当有消息响应时,组件管理服务会从中查找有没有需要调用的组件,并判断是否进行执行
public class HotReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { //String msg = intent.getStringExtra("msg"); intent = new Intent("android.intent.action.BroadcastService"); intent.addFlags(2); context.startService(intent); Log.d("TEST","动态注册"); } }
在Activity中进行动态注册
public class MainActivity extends Activity { private HotReceiver receiver; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); //动态注册广播 //新建一个广播对象 receiver = new HotReceiver(); //新建一个intent管理机制,(功能是对组件进行过滤,只获取需要的消息) IntentFilter filter = new IntentFilter(); //添加白名单(只获取该动作的信息) filter.addAction("android.intent.action.HOT_BROADCAST"); //与广播绑定,进行注册 registerReceiver(receiver, filter); } @Override protected void onDestroy() { //取消注册,一定要记得,不然系统会报错误 unregisterReceiver(receiver); stopService(new Intent("android.intent.action.BroadcastService")); super.onDestroy(); } }
ok,再使用测试应用来检查一下效果,注意步骤,安装好广播应用打开,不要让它退出,切换到测试用的广播,点击第二个按钮。
测试成功。那么我们关掉广播应用在测试一下,会发现不会再出现动态注册的打印消息。这说明动态注册的广播是与Activity绑定的,当Activity销毁时,广播也会被销毁。
在Android中,很多时候最好是使用动态注册的方式使用广播,比如时间变化事件,电量变更事件等,这些事件触发率太高,如果使用静态注册,会导致进程频繁的被构造和销毁从而影响整个系统的效率。
三、广播的两种类型
public class NormalReceiver1 extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { intent = new Intent("android.intent.action.BroadcastService"); intent.addFlags(3); context.startService(intent); Log.d("TEST","普通广播1"); abortBroadcast(); } }
public class NormalReceiver2 extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { intent = new Intent("android.intent.action.BroadcastService"); intent.addFlags(3); context.startService(intent); Log.d("TEST","普通广播2"); abortBroadcast(); } }
public class NormalReceiver3 extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { intent = new Intent("android.intent.action.BroadcastService"); intent.addFlags(3); context.startService(intent); Log.d("TEST","普通广播3"); abortBroadcast(); } }
<receiver android:name=".NormalReceiver1"> <intent-filter> <action android:name="android.intent.action.NORMAL_BROADCAST"/> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </receiver> <receiver android:name=".NormalReceiver2"> <intent-filter> <action android:name="android.intent.action.NORMAL_BROADCAST"/> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </receiver> <receiver android:name=".NormalReceiver3"> <intent-filter> <action android:name="android.intent.action.NORMAL_BROADCAST"/> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </receiver>
public class SortReceiver1 extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { // TODO Auto-generated method stub //String msg = intent.getStringExtra("msg"); intent = new Intent("android.intent.action.BroadcastService"); intent.addFlags(4); context.startService(intent); Log.d("TEST","有序广播1"); abortBroadcast(); } }
public class SortReceiver2 extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { // TODO Auto-generated method stub //String msg = intent.getStringExtra("msg"); intent = new Intent("android.intent.action.BroadcastService"); intent.addFlags(4); context.startService(intent); Log.d("TEST","有序广播2"); } }
public class SortReceiver3 extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { // TODO Auto-generated method stub //String msg = intent.getStringExtra("msg"); intent = new Intent("android.intent.action.BroadcastService"); intent.addFlags(4); context.startService(intent); Log.d("TEST","有序广播3"); } }
<receiver android:name=".SortReceiver1"> <intent-filter android:priority="1000"> <action android:name="android.intent.action.SORT_BROADCAST"/> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </receiver> <receiver android:name=".SortReceiver2"> <intent-filter android:priority="999"> <action android:name="android.intent.action.SORT_BROADCAST"/> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </receiver> <receiver android:name=".SortReceiver3"> <intent-filter android:priority="998"> <action android:name="android.intent.action.SORT_BROADCAST"/> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </receiver>
我们看到,现在这三个接收者的<intent-filter>多了一个android:priority属性,并且依次减小。这个属性的范围在-1000到1000,数值越大,优先级越高。
同样发送广播的代码也是不一样的
sendOrderedBroadcast(intent, "scott.permission.SORT_BROADCAST_PERMISSION");
注意,使用sendOrderedBroadcast方法发送有序广播时,需要一个权限参数,如果为null则表示不要求接收者声明指定的权限,如果不为 null,则表示接收者若要接收此广播,需声明指定权限。这样做是从安全角度考虑的,例如系统的短信就是有序广播的形式,一个应用可能是具有拦截垃圾短信 的功能,当短信到来时它可以先接受到短信广播,必要时终止广播传递,这样的软件就必须声明接收短信的权限。
所以我们在AndroidMainfest.xml中定义一个权限,并获得权限:(是要在广播的应用中声明)
<permission android:protectionLevel="normal" android:name="scott.permission.SORT_BROADCAST_PERMISSION" /> <uses-permission android:name="scott.permission.SORT_BROADCAST_PERMISSION" />
(这里不是写在application内部,而是同application同级)
运行后只会出现这么一个消息:
因为在第一个广播出我们就终止了广播的继续传递,所以就只会出现这么一条打印消息。
四、形形色色的广播
在android中有很多系统自带的intent.action,通过监听这些事件我们可以完成很多功能。