《Android优化专题》——优化电池续航能力

《Android优化专题》——优化电池续航能力

一、监测设备的电量和充电状态

在用户充电时,程序做任何操作都不会太受到电量影响,此时就适合做一些下载,刷新数据等耗电操作。

1. 判断当前充电状态

BatteryManager会广播Sticky intent,我们不需要注册BroadcastReiver,

1
2
IntentFilter ifilter = new IntentFilter(Intent.ACTION_BATTERY_CHANGED);
Intent batteryStatus = context.registerReceiver(null, ifilter);
1
2
3
4
5
6
7
8
9
// Are we charging / charged?  
int status = batteryStatus.getIntExtra(BatteryManager.EXTRA_STATUS, -1);
boolean isCharging = status == BatteryManager.BATTERY_STATUS_CHARGING ||
status == BatteryManager.BATTERY_STATUS_FULL;

// How are we charging?
int chargePlug = battery.getIntExtra(BatteryManager.EXTRA_PLUGGED, -1);
boolean usbCharge = chargePlug == BATTERY_PLUGGED_USB;
boolean acCharge = chargePlug == BATTERY_PLUGGED_AC;

通过判断设备此时是不是通过AC充电器,USB充电,不在充电状态三种情况来进行是否耗电操作,通常做法是,在使用AC充电时最大化后台更新操作,在使用USB充电时降低更新操作,不在充电状态时,最小化更新操作。

2. 监测充电状态的改变

充电状态随时改变,通过检查充电状态的改变,来改变App的某些行为。我们需要在Manifest文件里面注册一个监听来接收ACTION_POWER_CONNECTED and ACTION_POWER_DISCONNECTED的intent。

1
2
3
4
5
6
<receiver android:name=".PowerConnectionReceiver">
<intent-filter>
<action android:name="android.intent.action.ACTION_POWER_CONNECTED"/>
<action android:name="android.intent.action.ACTION_POWER_DISCONNECTED"/>
</intent-filter>
</receiver>
1
2
3
4
5
6
7
8
9
10
11
12
public class PowerConnectionReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
int status = intent.getIntExtra(BatteryManager.EXTRA_STATUS, -1);
boolean isCharging = status == BatteryManager.BATTERY_STATUS_CHARGING ||
status == BatteryManager.BATTERY_STATUS_FULL;

int chargePlug = intent.getIntExtra(BatteryManager.EXTRA_PLUGGED, -1);
boolean usbCharge = chargePlug == BATTERY_PLUGGED_USB;
boolean acCharge = chargePlug == BATTERY_PLUGGED_AC;
}
}

3. 判断当前电池电量

对于一些情况,获取当前电量对于我们是否要进行某个级别的后台操作十分有意义。我们可以从获取电池状态的intent中提取电池电量和容量信息。

1
2
3
int level = battery.getIntExtra(BatteryManager.EXTRA_LEVEL, -1);
int scale = battery.getIntExtra(BatteryManager.EXTRA_SCALE, -1);
float batteryPct = level / (float)scale;

4. 检测电量的有效改变

检测电量的有效改变,包括电量进入低电量模式,充电后离开低电量模式,这两种状态的变更是值得我们关注的。我们仅仅需要监听ACTION_BATTERY_LOW与ACTION_BATTERY_OKAY.

1
2
3
4
5
6
<receiver android:name=".BatteryLevelReceiver">
<intent-filter>
<action android:name="android.intent.action.ACTION_BATTERY_LOW"/>
<action android:name="android.intent.action.ACTION_BATTERY_OKAY"/>
</intent-filter>
</receiver>

对于设备进入低电量模式我们要尤其注意,这个时候的任何更新、下载等操作是非常影响用户体验,尤其还有可能下载到一半设备就已经关机了。

二、判断设备的停驻模式

Android设备有好几种停驻状态,为设备充电,包括车载模式,家庭模式,数字对战模拟模式。停驻状态与充电状态是非常密切相关联的。

1. 判断当前停驻状态

停驻状态的广播内容是sticky intent,所以不需要注册广播

1
2
3
4
5
IntentFilter ifilter = new IntentFilter(Intent.ACTION_DOCK_EVENT);
Intent dockStatus = context.registerReceiver(null, ifilter);

int dockState = battery.getIntExtra(EXTRA_DOCK_STATE, -1);
boolean isDocked = dockState != Intent.EXTRA_DOCK_STATE_UNDOCKED;

2. 判断当前停驻类型

  • Car
  • Desk
  • Low-End (Analog) Desk:API level 11开始才有
  • High-End (Digital) Desk:API level 11开始才有
1
2
3
4
boolean isCar = dockState == EXTRA_DOCK_STATE_CAR;
boolean isDesk = dockState == EXTRA_DOCK_STATE_DESK ||
dockState == EXTRA_DOCK_STATE_LE_DESK ||
dockState == EXTRA_DOCK_STATE_HE_DESK;

3. 监测停驻状态或者类型改变

只需要像下面一样注册监听器

1
<action android:name="android.intent.action.ACTION_DOCK_EVENT"/>

三、判断并监测网络连接状态

通过网络的连接状况改变,相应改变app的行为,减少无谓的操作,从而延长设备的续航能力。

1. 判断当前是否有网络连接

检查是否有网络连接

1
2
3
4
5
ConnectivityManager cm =
(ConnectivityManager)context.getSystemService(Context.CONNECTIVITY_SERVICE);

NetworkInfo activeNetwork = cm.getActiveNetworkInfo();
boolean isConnected = activeNetwork.isConnectedOrConnecting();

2. 判断连接网络的类型

分为移动网络,WiMax,Wi-Fi,以太网等连接类型

1
boolean isWiFi = activeNetwork.getType() == ConnectivityManager.TYPE_WIFI;

使用移动网络比WI-FI代价更大,多数情况下,移动网络下减少一些数据的获取操作,在有WIFI的情况下才开始。

3. 监测网络连接的切换

在manifest文件中注册一个带有action的Receiver

1
<action android:name="android.net.conn.CONNECTIVITY_CHANGE"/>

当然我们也没必要不间断地监听网络改变,我们只需要在完成某件任务时而进行监测即可。

4. 切换是否开启这些Receivers来提高效率

通过使用PackageManager来切换任何一个在mainfest定义好的组件开启状态,可以使用下面的方法来开启或者关闭任何一个broadcast receiver:

1
2
3
4
5
6
7
ComponentName receiver = new ComponentName(context, myReceiver.class);

PackageManager pm = context.getPackageManager();

pm.setComponentEnabledSetting(receiver,
PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
PackageManager.DONT_KILL_APP)

如果判断到了网络连接已经断开,可以在这个时候关闭除了网络环境改变之外的所有Receivers