【Android成长之路】全局大喇叭——广播机制的浅谈(发送自定义广播)

前言:在前一篇文章中已经介绍过了,广播主要分为两种类型,标准广播和有序广播,在本节中我们就将通过实践的方式来看下这两种广播具体的区别。并且实践了通过广播接收器来接收系统广播。接下来,就要尝试如何在应用程序中发送自定义的广播了。


一.发送标准广播
1.新建一个BoardcastTest2项目,默认生成MainActicity。在发送广播之前,我们还是需要先定义一个广播接收器来准备接收此广播才行,不然发出去也是白发。因此新建一个MyBroadcastReceiver 继承自BroadcastReceiver,代码如下所示:

public class MyBroadcastReceiver extends BroadcastReceiver {
   @Override
   public void onReceive(Context context, Intent intent) {
      Toast.makeText(context, "received in MyBroadcastReceiver",
      Toast.LENGTH_SHORT).show();
   }
}

这里当MyBroadcastReceiver 收到自定义的广播时,就会弹出received in MyBroadcastReceiver的提示。

2.然后在AndroidManifest.xml 中对这个广播接收器进行注册:

    "true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        ...
        ".MyBroadcastReceiver">
            
                "com.example.broadcasttest2.MY_BROADCAST" />
            
        
    

可以看到, 这里让MyBroadcastReceiver 接收一条值为com.example.broadcasttest2.MY_BROADCAST 的广播,因此待会儿在发送广播的时候,我们就需要发出这样的一条广播。

3.接下来修改activity_main.xml 中的代码,如下所示:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="Send Broadcast" />

LinearLayout>

这里在布局文件中定义了一个按钮,用于作为发送广播的触发点。

3.然后修改MainActivity中的代码,如下所示:

public class MainActivity extends Activity {
   ……
   @Override
   protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.activity_main);
      Button button = (Button) findViewById(R.id.button);
      button.setOnClickListener(new OnClickListener() {
                @Override
                public void onClick(View v) {
                    Intent intent = new Intent("com.example.broadcasttest2.MY_BROADCAST");
                    //这里一定要注意啊,//action在这里与注册的时候要一模一样,在末尾多一个空格都不行,小编在实践一个东东的时候,就是因为在Intent的action中的末尾多加了一个空格,害我苦苦改了半天,正要放弃的时候偶然才发现的,这是血淋林的教训啊!!!
                    sendBroadcast(intent);
                }
      });
   }
   ……
}

可以看到,我们在按钮的点击事件里面加入了发送自定义广播的逻辑。首先构建出了一个Intent 对象,并把要发送的广播的值传入,然后调用了Context 的sendBroadcast()方法将广播发送出去,这样所有监com.example.broadcasttest2.MY_BROADCAST 这条广播的广播接收器就会收到消息。此时发出去的广播就是一条标准广播

注意:这里,不知道大家有没有想过一个问题,注册的action一定要写成代码中的那个样子吗?答案是不一定,就算只有MY_BROADCAST也行(至于其它的奇葩写法,大家可以自己去尝试一下←_←),不过那样写是为了规范,毕竟规范一点还是好些。

运行程序,并点击一下Send Broadcast 按钮,效果如图:
【Android成长之路】全局大喇叭——广播机制的浅谈(发送自定义广播)_第1张图片
这样我们就成功完成了发送自定义广播的功能。另外,由于广播是使用Intent 进行传递的,因此你还可以在Intent 中携带一些数据传递给广播接收器。

注意:
广播是一种可以跨进程的通信方式,这一点从前面接收系统广播的时候就可以看出来了。因此在我们应用程序内发出的广播,其他的应用程序应该也是可以收到的。为了验证这一点,我们需要再新建一个BroadcastTest3 项目。
1.将项目创建好之后,还需要在这个项目下定义一个广播接收器,用于接收上一小节中的自定义广播。新建AnotherBroadcastReceiver 继承自BroadcastReceiver,代码如下所示:

public class AnotherBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "received in AnotherBroadcastReceiver",
        Toast.LENGTH_SHORT).show();
    }
}

2.这里仍然是在广播接收器的onReceive() 方法中弹出了一段文本信息。然后在AndroidManifest.xml 中对这个广播接收器进行注册,代码如下所示:

"true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        ...

        ".AnotherBroadcastReceiver" >
            
                "com.example.broadcasttest2.MY_BROADCAST" />
            
        

可以看到, AnotherBroadcastReceiver 同样接收的是com.example.broadcasttest2.MY_BROADCAST 这条广播。现在运行BroadcastTest2 项目将这个程序安装到模拟器上,然后重新回到BroadcastTest 项目的主界面,并点击一下Send Broadcast 按钮,就会分别弹出两次提示信息,如下图所示:
【Android成长之路】全局大喇叭——广播机制的浅谈(发送自定义广播)_第2张图片
这样就强有力地证明了,我们的应用程序发出的广播是可以被其他的应用程序接收到的。

二.发送有序广播
到目前为止,程序里发出的都还是标准广播,现在我们来尝试一下发送有序广播。关闭BroadcastTest3 项目,然后修改MainActivity 中的代码,如下所示:

public class MainActivity extends Activity {
    ……
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button button = (Button) findViewById(R.id.button);
        button.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent("com.example.broadcasttest2.MY_BROADCAST");
                sendOrderedBroadcast(intent, null);
            }
        });
        ……
    }
    ……
}

可以看到, 发送有序广播只需要改动一行代码, 即将sendBroadcast() 方法改成sendOrderedBroadcast()方法。sendOrderedBroadcast()方法接收两个参数,第一个参数仍然是Intent,第二个参数是一个与权限相关的字符串,这里传入null 就行了。现在重新运行程序,并点击Send Broadcast 按钮,你会发现,两个应用程序仍然都可以接收到这条广播。看上去好像和标准广播没什么区别嘛,不过别忘了,这个时候的广播接收器是有先后顺序的,而且前面的广播接收器还可以将广播截断,以阻止其继续传播。那么该如何设定广播接收器的先后顺序呢?当然是在注册的时候进行设定的了,修改AndroidManifest.xml 中的代码,如下所示:

        <receiver
            android:name=".AnotherBroadcastReceiver"
            android:priority="100" >
            <intent-filter>
                <action android:name="com.example.broadcasttest.MY_BROADCAST" />
            intent-filter>
        receiver>

可以看到,我们通过android:priority 属性给广播接收器设置了优先级,优先级比较高的广播接收器就可以先收到广播。这里将MyBroadcastReceiver 的优先级设成了100,以保证它一定会AnotherBroadcastReceiver 之前收到广播。
既然已经获得了接收广播的优先权,那么MyBroadcastReceiver 就可以选择是否允许广播继续传递了。修改MyBroadcastReceiver 中的代码,如下所示:

public class MyBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Toast.makeText(context, "received in MyBroadcastReceive",
        Toast.LENGTH_SHORT).show();
        abortBroadcast();//阻止广播继续传递
    }
}

广播接收器将无法再接收到这条广播。现在重新运行程序,并点击一下Send Broadcast 按钮,你会发现, 只有MyBroadcastReceiver 中的Toast 信息能够弹出, 说明这条广播经过MyBroadcastReceiver 之后确实是终止传递了。

你可能感兴趣的:(Android学习)