前言
项目中用到了BroadcastReceiver监听网络状态的改变用以控制下载任务的暂停或继续,原来是在AndroidManifest中注册了一个Receiver,然后在Receiver中通过逻辑判断执行不同操作,本来是没有问题的,但这两天在看官方文档,发现这种方式在7.0版本及以上失效了,官方文档说明如下:
这样做的具体原因如下:
这里可以知道,当很多应用在AndroidManifest中注册同一个广播时,系统会加载所有以这种方式注册了这个广播的app,这样会对设备性能和用户体验造成重大影响。所以为了解决这个问题,最好不用这种注册方式。监听用户网络切换应该是每个app都会干的事,所以谷歌直接就禁止了在AndroidManifest中注册这个广播。既然如此,就来复习一下创建广播接收器的相关知识。
广播接收器的定义
个人理解广播接收器就类似观察者模式,当系统或者其他app发出一个广播的时候,只要你的app对这个消息感兴趣的话(注册了这个广播)就会收到这个消息,然后你就可以对这个消息进行处理,但如何注册广播并接收消息呢?下面介绍两种方法(这里以接收网络状态切换为例):
****注意事项****
由于这里要监听网络状态,所以首先必须加上相关权限
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
-
****动态注册法****
1,新建ConnectivityReceiver类继承BroadcastReceiver类,覆盖onReceive方法,写入相关逻辑
2,在相关Activity中新建ConnectivityReceiver对象并初始化,新建IntentFilter对象并初始化,设置IntentFilter的Action
3,使用registerReceiver(BroadcastReceiver,IntentFilter)方法注册广播。
public class MainActivity extends AppCompatActivity{... private ConnectivityReceiver connectivityReceiver; private void registerNetReceiver(){ connectivityReceiver = new ConnectivityReceiver(); IntentFilter intentFilter = new IntentFilter(); intentFilter.addAction(ConnectivityManager.CONNECTIVITY_ACTION); registerReceiver(connectivityReceiver,intentFilter); } ... }
4,如果在onStart中注册的,在onStop中记得反注册,如果在onCreate中注册的,记得在onDestroy中反注册,不然会内存泄漏,反注册调用unregisterReceier(BroadcastReceiver)就行
-
****AndroidManifest.xml注册法(个人感觉非必要的话最好使用动态注册的方式)****
步骤
1,新建ConnectivityReceiver类继承BroadcastReceiver类,覆盖onReceive方法
2,在AndroidManifest中注册<receiver></receiver>节点,exported属性如果为true表示接收器接收除本应用(准确来说是同一个进程)以外的应用发送过来的消息,如果为false表示接收器只接受本应用发送的消息
<receiver android:name=".ConnectivityReceiver" android:exported="false">
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>
</intent-filter>
</receiver>
3,在onReceiver方法中写入相关逻辑
public class ConnectivityReceiver extends BroadcastReceiver {@Override public void onReceive(Context context, Intent intent) { ConnectivityManager connectivityManager = (ConnectivityManager)context.getSystemService(Context.CONNECTIVITY_SERVICE); NetworkInfo info = connectivityManager.getActiveNetworkInfo(); if(info == null){ Toast.makeText(context, "no network", Toast.LENGTH_SHORT).show(); }else{ switch (info.getType()){ case ConnectivityManager.TYPE_WIFI: Toast.makeText(context, "wifi", Toast.LENGTH_SHORT).show(); break; case ConnectivityManager.TYPE_MOBILE: Toast.makeText(context, "mobile", Toast.LENGTH_SHORT).show(); break; } } } }