美文网首页Android各类基础知识
Broadcast Receiver广播接收者详细解析

Broadcast Receiver广播接收者详细解析

作者: 付凯强 | 来源:发表于2017-10-31 11:57 被阅读385次

    1. 广播机制简介

    Android的广播机制非常灵活,Android的每一个应用程序都可以对自己感兴趣的广播进行注册,这样该程序就只会接收到自己所关心的广播内容,这些广播可能是来自系统的,也可能是来自于其他应用程序的。
    Android提供了一套完整的API,允许应用程序自由地发送和接收广播。发送广播借助Intent,而接收广播的方法利用广播接收器Broadcast Receiver。

    2. 广播的分类

    • 标准广播
      一种完全异步执行的广播。在广播发出之后,所有的广播接收器几乎都会在同一时刻接收到这条广播信息,因此他们之间没有任何先后顺序可言。这种广播的效率会比较高,但同时也意味着它是无法被截断的。
    • 有序广播
      一种同步执行的广播,在广播发出之后,同一时刻只会有一个广播接收器能够收到这条广播信息,当这个广播接收器中的逻辑执行完毕后,广播才会继续传递。此时的广播是有先后顺序的,优先级别高的广播接收器就可以先收到广播信息,并且前面的广播接收器还可以截断正在传递的广播,这样后面的广播接收器就无法收到广播信息了。

    3. 接收系统广播

    Android 内置了很多系统级别的广播,我们可以在应用程序中通过监听这些广播来得到各种系统的状态信息。比如手机开机完成后会发出一条广播,电池的电量发生变化会发出一条广播,时间或时区发生改变也会发出一条广播等。而想要接收到这些广播,需要使用广播接收器。

    动态注册监听网络变化

    • 注册广播的分类
      动态注册:代码中注册。
      静态注册:AndroidManifest.xml注册
    • 创建广播接收器
      只需新建一个类,让它继承自Broadcast-Receiver,并重写父类的onReceive方法即可。这样广播到来时,onReceive方法就会得到执行,具体的逻辑就可以在这个方法中处理。
    • 新建BroadcastTest项目,然后修改MainActivity中的代码:
    public class MainActivity extends AppCompatActivity {
    
    
        private IntentFilter intentFilter;
        private NetworkChangeReceiver networkChangeReceiver;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            intentFilter = new IntentFilter();
            intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
            networkChangeReceiver = new NetworkChangeReceiver();
            registerReceiver(networkChangeReceiver, intentFilter);
        }
    
        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);
        }
    }
    
    1. 创建广播过滤器 new IntentFilter() ,添加一个值为android.net.conn.CONNECTIVIT_CHANGE的action,之所以添加这个值,是因为当网络状态发生变化的时候,系统发出的正是一条值为android.net.conn.CONNECTIVITY_CHANGE的广播。
    2. 创建广播接收者NetworkChangeReceiver,并重写onReceive方法,每当网络状态发生变化的时候,onReceive方法就会得到执行,这里只是简单地使用Toast提示一段文本信息。
    3. 调用registerReceiver方法进行注册,将广播接收者和过滤者的实例都传进去,就可以监听网络变化的状态了。
    4. 动态注册的广播接收器一定要取消注册才行:在onDestory()方法中调用ungisterReceiver()方法来实现。
    5. 首先注册完成的时候会收到一条广播,然后修改下网络状态,又会受到一条广播,就 TOast提醒网络状态发生了变化。
    6. 只是提醒网络是否发生变化并不太人性化,还要准确地告诉用户当前是有网络还是没有网络,因此我们还需要对上面的代码进行优化,修改onReceive中的代码:
    class NetworkChangeReceiver extends BroadcastReceiver {
    
            @Override
            public void onReceive(Context context, Intent intent) {
                ConnectivityManager connectivityManager = (ConnectivityManager)
                getSystemService(Context.CONNECTIVITY_SERVICE);
                NetworkInfo networkInfo = connectivityManager.getActiveNetworkInfo();
                if (networkInfo!=null&&networkInfo.isAvailable()){
                    Toast.makeText(context,"network is available",Toast.LENGTH_SHORT).
                    show();
                }else{
                    Toast.makeText(context,"network is unavailable",Toast.LENGTH_SHORT).
                    show();
                }
            }
        }
    
    1. 在onReceive方法中,首先通过getSystemService方法得到ConnectivityManager的 实例,这是系统服务类,专门用于管理网络连接的。
    2. 然后调用它的getActiveNetworkInfo方法可以得到NetworkInfo的实例。
    3. 接着调用NetworkInfo的isAvailable方法,就可以判断出当前是否有网络了。
    4. 添加访问网络状态的权限(6.0就要动态注册了,后面讲):
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    

    静态注册实现开机启动

    动态注册的广播接收器虽然可以自由地控制注册和注销,但是必须在程序启动之后才能接收到广播。如果想让程序在未启动的情况下就能接收到广播,就需要静态注册了。现在我们让程序接收一条开机广播,当收到这个条广播时就可以在onReceive()方法里执行相应的逻辑,从而实现开机启动的功能。

    • 使用AS提供快捷方式来创建一个广播接收器,右击com.example.broadcasttest包——New——Ohter——Broadcast Receiver,会弹出一个窗口,在Class Name里面输入广播接收器的名字BootCompleteReceiver,Exported属性表示是否允许这个广播接收器接收本程序以外的广播,Enabled属性表示是否启用这个广播接收器,勾选这两个属性,点击Finish完成创建。
    • 修改BootCompleteReceiver中的代码:
    public class BootCompleteReceiver extends BroadcastReceiver {
    
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"Boot Complete",Toast.LENGTH_LONG).show();
        }
    }
    
    • 静态的广播接收器一定要在AndroidManifest.xml中注册才可以使用,不过由于先前是使用AS快捷方式创建的广播接收器,因此注册这一步已经被自动完成了。会发现application标签内多了如下代码:
        <receiver
        android:name=".BootCompleteReceiver"
        android:enabled="true"
        android:exported="true"></receiver>
    

    *** 说明:在<application>标签内出现了一个新的标签<receiver>,所有静态的广播接收器都是在这里进行注册的。它的用法其实和<activity>标签很相似,都是通过android:name来指定具体注册哪一个广播接收器,而 enaled和exported属性则是根据我们刚才勾选的状态自动生成的。

            <receiver
                android:name=".BootCompleteReceiver"
                android:enabled="true"
                android:exported="true"></receiver>
    
    • 不过目前BootCompleteReceiver还是不能接收到开机广播的,还需要对广播进行限定,添加广播过滤器和申请权限。
            <receiver
                android:name=".BootCompleteReceiver"
                android:enabled="true"
                android:exported="true">
                <intent-filter>
                    <action android:name="android.intent.action.BOOT_COMPLETED" />
                </intent-filter>
            </receiver>
    
    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
    
    1. 由于Android系统启动完成后会发出一条值为android.permission.RECEIVE_BOOT_COMPLETED的广播,因此我们 <intent-filter>标签里添加了相应的action。
    2. 监听系统开机广播也是需要声明权限的,我们使用<uses-permission>标签又加入一条android.permission.RECEIVE_BOOT_COMPLETED权限。
    3. 将模拟器重新启动就可以收到开机广播了。
    4. 需要额外注意的是,不要在onReceive()方法中添加过多的逻辑或者进行任何的耗时操作,因为在广播接收器中是不允许开启线程的,当onReceive方法运行较长时间而没有结束时,程序就会报错。所以广播接收器更多的是扮演一种打开程序其他组件的角色,比如创建一条状态栏通知,或者启动一个服务等。

    4. 发送自定义广播

    以上学习的是系统广播,现在我们学习如何在程序中发送自定义广播。广播分为标准广播和有序广播,本节我们就将通过实践的方式来看一下这两种广播具体的区别。

    发送标准广播

    • 定义一个广播接收器,新建一个MyBroadcastReceiver,代码:
    public class MyBroadcastReceiver extends BroadcastReceiver {
    
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"received in MyBroadcastReceiver",Toast.LENGTH_SHORT).
            show();
        }
    }
    
    • 以上是通过快捷方式创建的广播接收器,所以配置文件里面已经有了receiver标签,修改标签内容,即添加过滤者,代码如:
    <receiver
        android:name=".MyBroadcastReceiver"
        android:enabled="true"
        android:exported="true">
        <intent-filter>
            <action android:name="com.beidou.broadcasttest.MY_BROADCAST">
            </action>
        </intent-filter>
    </receiver>
    

    *** 这里让MyBroadcastReceiver接收一条值为com.beidou.broadcasttest.MY_BROADCAST的广播,因此待会在发送广播的时候,我们需要发出这样的一条广播。

    • 修改activity_main.xml以及MainActivity中的代码:
    Button button = (Button) findViewById(R.id.button);
    button.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            Intent intent = new Intent("com.beidou.broadcasttest.MY_BROADCAST");
            sendBroadcast(intent);
        }
    });
    
    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="send broadcast!" />
    

    *** 在点击事件中,我们加入了发送自定义广播的逻辑。

    1. 首先构建出了一个Intent对象,并把要发送的广播的值传入。
    2. 然后调用了Context的sendBroadcast方法将广播发送出去,这样所有监听com.com.beidou.broadcasttest.MY_BROADCAST这条广播的广播接收器就会受到信息。
    3. 由于广播是使用Intent进行传递的,因此你还可以在Intent中携带数据传递给广播接收器。

    发送有序广播

    • 验证其他的应用程序之间可以收到广播
    1. 新创建 BroadcastTest2 项目,点击AS-File-New-New Project进行创建.
    2. 新建AnotherBroadcastReceiver,代码:
    public class AnotherBroadcastReceiver extends BroadcastReceiver {
    
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"received in AnotherBroadcastReceiver",
            Toast.LENGTH_SHORT).show();
        }
    }
    
    1. 修改配置文件
    <receiver
        android:name=".AnotherBroadcastReceiver"
        android:enabled="true"
        android:exported="true">
        <intent-filter>
            <action android:name="com.beidou.broadcasttest.MY_BROADCAST"></action>
        </intent-filter>
    </receiver>
    
    1. 证明了应用程序发出的广播是可以被其他的程序接收到的。
    • 发送有序广播,修改MainActivity代码:
    Intent intent = new Intent("com.beidou.broadcasttest.MY_BROADCAST");
    sendOrderedBroadcast(intent,null);
    
    1. 发送有序广播只需改动一行代码,即将sendBroadcast()方法改成sendOrderedBroadcast方法。接收两个参数,第一个参数仍然是Intent,第二个参数是一个与权限相关的字符串,这里传入null即可。
    2. 重新运行程序,两个应用程序都可以接收到这条广播,但是这个时候的广播接收器是有先后顺序的,而且前面的广播接收器还可以将广播截断,以阻止其继续广播。
    3. 设定广播接收器的先后顺序,在配置文件里面
    <receiver
        android:name=".MyBroadcastReceiver"
        android:enabled="true"
        android:exported="true">
        <intent-filter android:priority="100">
            <action android:name="com.beidou.broadcasttest.MY_BROADCAST"></action>
        </intent-filter>
    </receiver>
    

    *** 通过android:priority="100",给MyBroadcastReceiver的优先级设成了100,以保证它一定会在AnotherBroadcastReceiver之前收到广播。

    1. MyBroadcastReceiver获得了接收广播的优先权,那么MyBroadcastReceiver就可以选择是否允许广播继续传递了。修改MyBroadcastReceiver代码:
    public class MyBroadcastReceiver extends BroadcastReceiver {
    
        @Override
        public void onReceive(Context context, Intent intent) {
            Toast.makeText(context,"received in MyBroadcastReceiver",Toast.LENGTH_LONG).
            show();
            abortBroadcast();
        }
    }
    

    *** 在onReceive方法中调用了abortBroadcast方法,就表示将这条广播截断,后面的广播接收器将无法再接收到这条广播。

    5. 使用本地广播

    前面我们发送和接收的广播全部属于系统全局广播,即发出的广播既可以被其他任何应用程序接收到,并且我们可以接收到来自于其他任何应用程序的广播。这样就存在安全性问题。

    比如:我们发送的一些携带关键性数据的广播有可能被其他应用程序截获,或者其他的程序不停地向我们的广播接收器里发送各种垃圾广播。

    为此,Android引入了一套本地广播机制:使用这个广播机制发出的广播只能够在应用程序的内部进行传递,并且广播接收器也只能接收来自本应用程序发出的广播。

    • 用法:主要是使用了一个LocalBroadcastManager来对广播进行管理,并提供了发送广播和注册广播接收器的方法。
    • 修改MainActivity的代码:
    public class MainActivity extends AppCompatActivity {
    
    
        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 View.OnClickListener() {
                @Override
                public void onClick(View view) {
                    Intent intent = new Intent("com.beidou.broadcasttest.LOCAL_BROADCAST");
                    localBroadcastManager.sendBroadcast(intent);//发送广播
                }
            });
    
            intentFilter = new IntentFilter();
            intentFilter.addAction("com.beidou.broadcasttest.LOCAL_BROADCAST");
            localReceiver = new LocalReceiver();
            localBroadcastManager.registerReceiver(localReceiver, intentFilter);//注册广播
    
        }
    
        class LocalReceiver extends BroadcastReceiver {
    
            @Override
            public void onReceive(Context context, Intent intent) {
                Toast.makeText(context, "received local broadcast", Toast.LENGTH_SHORT).
                show();
            }
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            localBroadcastManager.unregisterReceiver(localReceiver);//注销广播
        }
    }
    
    
    1. 以上代码和动态注册广播接收器以及发送广播的代码是一样的。
    2. 只不过现在首先是通过LocalBroadcastManager的getInstance()方法获取它的一个实例,然后在注册广播接收器的时候调用的是LocalBroadcastManager的registerReceiver()方法,在发送广播的时候调用的是LocalBroadcastManager的sendBroadcast()方法。
    3. 注意点:本地广播是无法静态注册的方式来接收的。主要是因为静态注册主要就是为了让程序在未启动的情况下也能收到广播,而发送本地广播时,我们的程度肯定是已经启动的,因此也 不需要使用静态注册的功能。
    4. 本地广播的优势:
      ① 可以明确地知道正在发送的广播不会离开我们的程序,因此不必担心泄密。
      ② 其他的程序无法将广播发送到我们程序的内部,因此不需要担心会有安全漏洞。
      ③ 发送本地广播比发送系统广播将会更加高效。

    6. 广播的最佳实践——实现强制下线功能

    强制下线功能比较常见:比如QQ在别处登录了,就会将你强制挤下线。

    逻辑:只需要在界面上弹出一个对话框,让用户无法进行任何其他操作,必须要点击对话框中的确定按钮,然后回到登录界面即可。

    疑问:当我们被通知需要强制下线时可能正处于任何一个界面,难道需要在每个界面上都编写一个弹出对话框的逻辑?那到不用,借助广播知识,可以轻松实现这一个功能。

    • 强制下线功能需要先关闭掉所有的活动,然后回到登录界面。
    • 创建BroadcastBestPractice项目,创建一个ActivityCollector类用于管理所有的活动:
    public class ActivityCollector {
        public static List<Activity> activities = new ArrayList<>();
    
        public static void addActivity(Activity activity) {
            activities.add(activity);
        }
    
        public static void removeActivity(Activity activity) {
            activities.remove(activity);
        }
    
        public static void finishAll() {
            for (Activity activity : activities) {
                if (!activity.isFinishing()) {
                    activity.finish();
                }
            }
        }
    }
    
    
    • 创建BaseActivity类作为所有活动的父类。
    public class BaseActivity extends AppCompatActivity {
        
        @Override
        protected void onCreate(@Nullable Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            ActivityCollector.addActivity(this);
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            ActivityCollector.removeActivity(this);
        }
    }
    
    • 创建一个登录页面的活动,新建LoginActivity,编辑布局activity_login.xml:
    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">
    
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="60dp"
            android:orientation="horizontal">
    
            <TextView
                android:layout_width="90dp"
                android:layout_height="wrap_content"
                android:layout_gravity="center_vertical"
                android:text="Account:"
                android:textSize="18sp" />
    
            <EditText
                android:id="@+id/account"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_gravity="center_vertical"
                android:layout_weight="1" />
        </LinearLayout>
    
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="60dp"
            android:orientation="horizontal">
    
            <TextView
                android:layout_width="90dp"
                android:layout_height="wrap_content"
                android:layout_gravity="center_vertical"
                android:text="Password:"
                android:textSize="18sp" />
    
            <EditText
                android:id="@+id/password"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_gravity="center_vertical"
                android:layout_weight="1"
                android:inputType="textPassword" />
        </LinearLayout>
    
        <Button
            android:id="@+id/login"
            android:layout_width="match_parent"
            android:layout_height="60dp"
            android:text="login" />
    </LinearLayout>
    
    public class LoginActivity extends BaseActivity {
    
        private EditText accountEdit;
        private EditText passwordEdit;
        private Button login;
    
        @Override
        protected void onCreate(@Nullable Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_login);
            accountEdit = (EditText) findViewById(R.id.account);
            passwordEdit = (EditText) findViewById(R.id.password);
            login = (Button) findViewById(R.id.login);
            login.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View view) {
                    String account = accountEdit.getText().toString();
                    String password = passwordEdit.getText().toString();
                    //如果账号是admin 且密码是123456,就认为登录成功
                    if (account.equals("admin") && password.equals("123456")) {
                        startActivity(new Intent(LoginActivity.this, MainActivity.class));
                        finish();
                    } else {
                        Toast.makeText(LoginActivity.this, 
                        "account or password is invalid", Toast.LENGTH_LONG).show();
                    }
                }
            });
        }
    }
    
    
    • 修改activity_main.xml和MainActivity:
    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical">
    
        <Button
            android:id="@+id/force_offline"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="Send force offline broadcast" />
    </LinearLayout>
    
    public class MainActivity extends BaseActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            Button forceoffline = (Button) findViewById(R.id.force_offline);
            forceoffline.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View view) {
                    Intent intent = new Intent("com.beidou.broadcastbestpractice.
                    FORCE_OFFLINE");
                    sendBroadcast(intent);
                }
            });
        }
    }
    
    • 在BaseActivity中动态注册一个广播接收器就可以了。
    public class BaseActivity extends AppCompatActivity {
        private ForceOfflineReceiver receiver;
    
        @Override
        protected void onCreate(@Nullable Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            ActivityCollector.addActivity(this);
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            ActivityCollector.removeActivity(this);
        }
    
        @Override
        protected void onResume() {
            super.onResume();
            IntentFilter intentFilter = new IntentFilter();
            intentFilter.addAction("com.beidou.broadcastbestpractice.FORCE_OFFLINE");
            receiver = new ForceOfflineReceiver();
            registerReceiver(receiver,intentFilter);
        }
    
        @Override
        protected void onPause() {
            super.onPause();
            if (receiver!=null){
                unregisterReceiver(receiver);
                receiver = null;
            }
        }
    
        class ForceOfflineReceiver extends BroadcastReceiver {
    
            @Override
            public void onReceive(final Context context, Intent intent) {
                AlertDialog.Builder builder = new AlertDialog.Builder(context);
                builder.setTitle("Warning");
                builder.setMessage("You are forced to be offline.
                Please try to login again");
                builder.setCancelable(false);
                builder.setPositiveButton("OK", new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialogInterface, int i) {
                        ActivityCollector.finishAll(); //销毁所有的活动
                        Intent intent = new Intent(context,LoginActivity.class);
                        context.startActivity(intent);//重新启动LoginActivity
                    }
                });
                builder.show();
            }
        }
    }
    
    • 修改AndroidManifest.xml文件:
    <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"></activity>
            <activity android:name=".LoginActivity">
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
    
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
        </application>
    

    ***** 综上:当账号和密码正确的时候调转到主页面MainActivity,在主页面里面发出一个广播(BaseActivity里面已经加入了注册和取消广播的代码),所以广播发出后,BaseActivity里面的广播接收器就会接收到广播,与此同时弹出对话框,点击OK,进行强制所有活动下线的操作,并重新打开登录页面。

    7. Git-初识版本控制工具

    安装Git(百度下,安装下即可,版本号2.8.1)

    打开Git Bash

    创建代码仓库

    1. 配置身份
      git config --global user.name "FUkaiqiang"
      git config --global user.email "18201685396@163.com"
    2. 验证身份
      git config --global user.name
      git config --global user.email
    3. 创建代码仓库(进入项目存放的目录)
      cd D:
      cd Test
      cd BroadcastBestPractice
      git init

    *** 此时会生成一个.git文件夹,用来记录所有的Git操作的。

    1. 查看列表
      ls -al
    2. 提交本地代码(先用add把想要的代码先添加进来,而conmit则是真正地去执行提交操作)
      提交单个文件 git add build.gradle
      提交整个目录 git add .
      提交:git commit -m "First commit" (m不可少)

    相关文章

      网友评论

        本文标题:Broadcast Receiver广播接收者详细解析

        本文链接:https://www.haomeiwen.com/subject/ptqbpxtx.html