BroadcastReceiver与自定义权限

来源:互联网 发布:深圳悍马军品淘宝店 编辑:程序博客网 时间:2024/06/06 10:43

自定义权限

        在清单文件中通过<permission>进行声明,然后通过<uses-permission>引用自己声明的权限(可不引用)。示例:

    <!-- 引用自己声明的权限 -->    <uses-permission android:name="com.baigle.imitatehandler.TEST" /><!-- 声明自定义权限 -->    <permission        android:name="com.baigle.imitatehandler.TEST"        android:protectionLevel="signature" />

protectionLevel

        在自定义权限时,通常会指定protectionLevel属性,常用的如下:

        normal:默认的,应用安装前,用户可以看到相应的权限,但无需用户主动授权。

        dangerous:normal安全级别控制以外的任何危险操作。需要dangerous级别权限时,Android会明确要求用户进行授权。常见的如:网络使用权限,相机使用权限及联系人信息使用权限等。

        signature:它要求权限声明应用和权限使用应用使用相同的keystore进行签名。如果使用同一keystore,则该权限由系统授予,否则系统会拒绝。并且权限授予时,不会通知用户。它常用于应用内部。例如:

    <!-- 引用自己声明的权限 ,可不引用-->    <uses-permission android:name="com.baigle.imitatehandler.TEST" />    <!-- 声明自定义权限 -->    <permission        android:name="com.baigle.imitatehandler.TEST"        android:protectionLevel="signature" />        <receiver            android:name="com.baigle.imitatehandler.OtherActivity"            android:label="@string/app_name"            android:permission="com.baigle.imitatehandler.TEST" >            <intent-filter>                <action android:name="action1" />                <category android:name="android.intent.category.DEFAULT" />            </intent-filter>        </receiver>
        上面把protectionLevel声明为signature。如果别的应用使用的不是同一个签名文件,就没办法使用该权限,从而保护了自己的接收者。

BroadcastReceiver

        对于广播接收者来说,别的应用也可以监听并触发我们的广播接收者。如果广播接收者注册在清单文件中,只需要在<receiver>中配置android:exported="false"属性。这样,系统中的其它应用就无法触及到该receiver了。

        但如果receiver是动态注册的,就需要创建自己的使用权限,并且将protectionLevel设置为signature。这样,当别的应用和receiver所在的应用使用的签名不一样时,便不会启动该receiver。例如:

注册广播接收者

BroadcastReceiver receiver = new OtherActivity();IntentFilter filter = new IntentFilter("action1");filter.addCategory(Intent.CATEGORY_DEFAULT);//注册receiver时,直接指定发送者应该具有的权限。不然外部应用依旧可以触及到receiverregisterReceiver(receiver, filter, permission, null);
        在注册的时候,最关键的一点是用registerReceiver(BroadcastReceiver, IntentFilter, String, Handler)进行注册,而不是平常用的是registerReceiver(BroadcastReceiver, IntentFilter)。相较于后者,前者在注册的时候要求了发送者必须具有的权限。如果发送者没有该权限,那么发送者发送的广播即使经过IntentFilter的过滤,也不会被receiver接收。此时如果再自定义一个权限,并且将权限的protectionLevel设置为signature,那么外部应用便无法使用该权限,也就无法触及到该receiver。上面的permission便是这样的一个权限,声明如下:

    <!-- 引用自己声明的权限 -->    <uses-permission android:name="com.baigle.imitatehandler.TEST" />    <!-- 声明自定义权限 -->    <permission        android:name="com.baigle.imitatehandler.TEST"        android:protectionLevel="signature" />
        发送广播的代码如下:
Intent intent = new Intent("action1");intent.putExtra("text", "receiver");intent.addCategory(Intent.CATEGORY_DEFAULT);// 也可以使用sendBroadcast(intent);进行发送sendBroadcast(intent, permission);

有序广播

        通过sendBroadcast发送的广播都是无序广播,所有receiver的接收顺序和执行顺序都是无法确定,它们什么时候能运行完毕也是无法确定,同时也无法在receiver之间进行通信。从而也无法使用getResultCode(),setResult()及abortBroadcast()一系列的方法。
        通过sendOrderedBroadcast()发送的广播便是有序广播。

特点

有序广播有如下特点:
        1,所有的receiver依次执行优先级高低进行排序,优先级高的receiver先执行,直到最后。但优先级相同的receiver执行顺序不确定
        2,高优先级的receiver可以使用setResult()等方法向其后的receiver传递数据。
        3,低优先级的receiver可以通过getResult()等方法来获取高优先级receiver通过setResult()等方法传递的数据。
        4,所有的receiver都可以调用abortBroadcast()等方法中止广播,使广播不再往比它优先级低的receiver上传递。 
        5,优先级的注册方法有两种,在清单文件中可以通过为intent-filter中添加android:priority属性。在代码中,可以通过IntentFilter.setPriority()进行设置。优先级最低是-999

功能

        利用有序广播的上述特点,可以实现一些功能。比如当界面显示时不执行某个操作,而当界面不显示时执行某个操作。
        具体思路:将执行操作放在一个低优先级的receiver中,同时在界面中注册一个高优先级的receiver,并且两个receiver的intent一样。当界面接收到广播时,直接取消广播或者通知低优先的广播不执行操作。这里最主要的就是界面中的广播接收者要放在哪个方法中进行注册,一般是放在onStart()和onStop()中的。因为这两个方法确定了界面可见的边界
        注册和取消注册的代码略,只是在注册时要通过IntentFilter.setPriority()方法设置界面receiver的优先级,一般设置成1000。具体的接收者如下:
private class InterceptReceiver extends BroadcastReceiver {//界面显示时,如果有广播,会执行到该方法。public void onReceive(Context context, Intent intent) {//setResultCode(Activity.RESULT_CANCELED);//改变resultCode的值abortBroadcast();//或者直接取消广播}}
        执行操作的receiver一般要根据resultCode的值判断是否应该进行某种操作。如下:
public void onReceive(Context context, Intent intent) {int code = getResultCode();if (code == Activity.RESULT_OK) {//do somethingSystem.out.println("non canceled");} else {//do nothingSystem.out.println("canceled");}}
        发送广播也没什么问题,只是在发送的时候将resultCode的值初始值设置为Activity.RESULT_OK,这主要是为了和执行操作的receiver相对应。如:
// 设置resultCode的初始值为RESULT_OKsendOrderedBroadcast(intent, null, null, null,Activity.RESULT_OK, null, null);










0 0
原创粉丝点击