您的位置:首页 > 移动开发 > Android开发

Android四大组件——自定义广播

2017-03-29 17:57 295 查看
实现的效果是在ceshi7 中点击按钮,两个广播接收器都会收到广播。在AndroidManifest.xml文件中静态注册的Receiver不仅仅需要添加对应的action,还需要添加

<category android:name="android.intent.category.DEFAULT"/>。


话不多说,直接贴代码

ceshi7项目下的。

public class MainActivity extends AppCompatActivity {

private Button sendBroad;
private MySecondReceiver mySecondReceiver = new MySecondReceiver();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction("com.example.luzhao.ceshi.Broadcast");
registerReceiver(mySecondReceiver,intentFilter);
sendBroad = (Button)findViewById(R.id.sendBroadcast);
sendBroad.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Intent intent = new Intent("com.example.luzhao.ceshi.Broadcast");
sendBroadcast(intent);
}
});
}

@Override
protected void onDestroy() {
super.onDestroy();
unregisterReceiver(mySecondReceiver);
}

class MySecondReceiver extends BroadcastReceiver{

@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"广播测试",Toast.LENGTH_SHORT).show();
}
}
}

ceshi8项目下的

public class MyReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context,"ceshi8接收到广播",Toast.LENGTH_SHORT).show();
Log.i("MyReceiver","ceshi8");
}
}

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.luzhao.ceshi8">

<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />

<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<receiver android:name=".MyReceiver">
<intent-filter>
<action android:name="com.example.luzhao.ceshi.Broadcast"/>
<category android:name="android.intent.category.DEFAULT"/>
</intent-filter>
</receiver>
</application>

</manifest>

自定义权限

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

[html] view
plain copy

 





   <!-- 引用自己声明的权限 -->  

   <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,则该权限由系统授予,否则系统会拒绝。并且权限授予时,不会通知用户。它常用于应用内部。例如:

[html] view
plain copy

 





<!-- 引用自己声明的权限 ,可不引用-->  

<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。例如:
注册广播接收者

[java] view
plain copy

 





BroadcastReceiver receiver = new OtherActivity();  

IntentFilter filter = new IntentFilter("action1");  

filter.addCategory(Intent.CATEGORY_DEFAULT);  

//注册receiver时,直接指定发送者应该具有的权限。不然外部应用依旧可以触及到receiver  

registerReceiver(receiver, filter, permission, null);  

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

[html] view
plain copy

 





<!-- 引用自己声明的权限 -->  

<uses-permission android:name="com.baigle.imitatehandler.TEST" />  

<!-- 声明自定义权限 -->  

<permission  

    android:name="com.baigle.imitatehandler.TEST"  

    android:protectionLevel="signature" />  

        发送广播的代码如下:

[java] view
plain copy

 





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。具体的接收者如下:

[java] view
plain copy

 





private class InterceptReceiver extends BroadcastReceiver {  

    //界面显示时,如果有广播,会执行到该方法。  

    public void onReceive(Context context, Intent intent) {  

        //setResultCode(Activity.RESULT_CANCELED);//改变resultCode的值  

        abortBroadcast();//或者直接取消广播  

    }  

}  

        执行操作的receiver一般要根据resultCode的值判断是否应该进行某种操作。如下:

[java] view
plain copy

 





public void onReceive(Context context, Intent intent) {  

    int code = getResultCode();  

    if (code == Activity.RESULT_OK) {//do something  

        System.out.println("non canceled");  

    } else {//do nothing  

        System.out.println("canceled");  

    }  

}  

        发送广播也没什么问题,只是在发送的时候将resultCode的值初始值设置为Activity.RESULT_OK,这主要是为了和执行操作的receiver相对应。如:

[java] view
plain copy

 





// 设置resultCode的初始值为RESULT_OK  

sendOrderedBroadcast(intent, null, null, null,  

    Activity.RESULT_OK, null, null);
 
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息