【Android电池电量监控初步】

2023-05-16

引言

初学Andorid,学习了包括ActivityServiceAppWidgetBroadcastIntentNotificationXML布局、XML配置等基础知识之后,尝试着做一个小玩意儿——电池监控,期间也是阻挠多多,现在基本完成了,所以把自己的经验分享给大家,希望能通过我的文章让还不会的朋友少走弯路,也算是这个小玩意儿的一点贡献。

由于时间有限,bill假设读者已经具备并能基本使用上述基础知识,本文将只描述如何利用上述知识构建一个监控Android电池电量的小程序(若有相关知识不甚清楚,请查询Android SDK自带的document,官方文档是我们学习Android的一大利器),若有任何疑问或建议,欢迎留言提出,bill会很高兴和大家一起学习进步!

 

开发环境及Android平台版本

Eclipse helios

Android SDK 1.6

Android SDK 1.6 及以上模拟器或者Android SDK 1.6及以上Android X86 虚拟机或真机

 

功能描述

由于Android自带的系统电池图标并不能显示当前电量,给不少用户带来了不便,所以本程序将对此进行改进,简单实现对Android手机电量的实时监控,并在左上角状态栏显示实时电池电量的通知图标,若正在充电,则显示充电状态。为方便起见,还为本程序制作了一个AppWidget,成为一个简单的桌面小工具。

 

效果预览

充电效果

 

整体思路概览

      ①由于要显示电池图标,因此需要一组对应的小icon,程序算出当前电量之后,就在状态栏上显示对应的icon即可。本程序用到的icon已经提供在附件,下载后将里面的icons直接复制到res\drawable-hdpi中即可。

      ②由于电池更新是一个长期存在的过程,因此需要将我们的程序注册为系统服务,以便长期生存于Android系统中。

      ③服务不会自行启动,因此我们需要为自己的服务创建一个Activity,在第一次启动本程序的Activity时创建并启动我们的监控服务,之后的事便交由服务处理,Activity暂时未有太大作用(之后还可以在Activity中显示电池当前的各种基本属性等等,本文不作介绍),当然,我们也可以不要Activity,直接将我们的服务设置为开机启动即可,关于开机启动的事项已在【Android开机启动ActivityService的方法】一文中详细描述,恕不赘述。

      ④为了方便起见,应该制作本程序的桌面小工具,也就用到了AppWidget方面的知识,稍后一一道来。

 

分而治之

小麻雀的心脏——电池监控服务MonitorService

我们将要看到的是这个小麻雀的心脏——电池监控服务,本服务自启动至停止,始终通过接受来自系统的广播ACTION_BATTERY_CHANGED(本广播指示电量发生了改变)来判断并显示当前电量。

下面是本Service的完整代码,我会在可能费解的地方批注,希望能帮助大家理解。

 


  
  1. package com.billhoo.study;  
  2.  
  3. import android.app.Notification;  //通知,即显示在屏幕左上角的小图标  
  4. import android.app.NotificationManager;  
  5. import android.app.PendingIntent;  
  6. import android.app.Service;  
  7. import android.content.BroadcastReceiver;  
  8. import android.content.Context;  
  9. import android.content.Intent;  
  10. import android.content.IntentFilter;  
  11. import android.os.BatteryManager;  
  12. import android.os.IBinder;  
  13.  
  14. public class MonitorService extends Service {  
  15.   @Override 
  16.   public IBinder onBind(Intent intent) {  
  17.     return null;  
  18.   }  
  19.  
  20.   @Override 
  21.   public void onStart(Intent intent, int startId) {  
  22.     // 定义电池电量更新广播的过滤器,只接受带有ACTION_BATTERRY_CHANGED事件的Intent  
  23.     IntentFilter batteryChangedReceiverFilter = new IntentFilter();  
  24.     batteryChangedReceiverFilter.addAction(Intent.ACTION_BATTERY_CHANGED);  
  25.       
  26.     // 向系统注册batteryChangedReceiver接收器,本接收器的实现见代码字段处  
  27.     registerReceiver(batteryChangedReceiver, batteryChangedReceiverFilter);  
  28.  
  29.     // 实例化Notification通知的管理器,即字段notification manager  
  30.     notifyManager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);  
  31.  
  32.     // 由于初始化本服务时系统可能没有发出ACTION_BATTERY_CHANGED广播,那么刚才注册的那个接收器将不会在本服务启动时被激活,这种情况下就无法显示当前电量,因此在这里添加一个匿名广播接收器。  
  33.     new BroadcastReceiver() {  
  34.       @Override 
  35.       public void onReceive(Context context, Intent intent) {  
  36.         int level = intent.getIntExtra("level"0);  //电池电量等级  
  37.         int scale = intent.getIntExtra("scale"100);  //电池满时百分比  
  38.         int status = intent.getIntExtra("status"0);  //电池状态  
  39.  
  40.         // 若正在充电  
  41.         if (status == BatteryManager.BATTERY_STATUS_CHARGING)  
  42.           notification = getNotification(getChargingIcon(level * 100 / scale),  
  43.               "电池监控", System.currentTimeMillis(), "电池电量""正在充电");  
  44.         else 
  45.           notification = getNotification(getStateIcon(level * 100 / scale),  
  46.               "电池监控", System.currentTimeMillis(), "电池电量""请及时充电");  
  47.       
  48.     //向系统发送此通知,为方便起见,通知ID简单设为0  
  49.         notifyManager.notify(0, notification);  
  50.       }  
  51.     };  
  52.   }  
  53.  
  54.   @Override 
  55.   public void onDestroy() {  
  56.     //当本服务终止时,通过通知ID注销左上角的通知  
  57.     notifyManager.cancel(0);  
  58.   }  
  59.  
  60.   // --------------------------------------私有方法------------------------------  
  61.   // 获得一个notification  
  62.   private Notification getNotification(int notificationIcon, String tickerText,  
  63.       long when, String contentTitle, String contentText) {  
  64.     // 定义该notification的扩展文本和intent  
  65.     Notification notification = new Notification(notificationIcon, tickerText,  
  66.         when);  
  67.     Context context = getApplicationContext();  
  68.     Intent notificationIntent = new Intent(context, BatteryInfo.class);  
  69.     PendingIntent contentIntent = PendingIntent.getActivity(this0,  
  70.         notificationIntent, 0);  
  71.     notification.setLatestEventInfo(context, contentTitle, contentText,  
  72.         contentIntent);  
  73.  
  74.     // 设置本状态出现在“正在进行”而非“通知”栏目,不允许用户清除状态图标  
  75.     notification.flags |= Notification.FLAG_ONGOING_EVENT;  
  76.     return notification;  
  77.   }  
  78.  
  79.   // 获得对应的电池状态图标  
  80.   private int getStateIcon(int batteryHealth) {  
  81.     if (batteryHealth >= 0 || batteryHealth <= 100)  
  82.       return batteryStateIcons[batteryHealth]; //这里是为了方便起见而硬编码的资源数组,直接从R.java文件中提取的十六进制资源ID  
  83.     return R.drawable.stat_sys_battery_unknown;  
  84.   }  
  85.  
  86.   // 获得对应的充电状态图标  
  87.   private int getChargingIcon(int batteryHealth) {  
  88.     if (batteryHealth >= 0 && batteryHealth < 5)  
  89.       return R.drawable.stat_sys_battery_charge_anim0;  
  90.     if (batteryHealth >= 5 && batteryHealth < 10)  
  91.       return R.drawable.stat_sys_battery_charge_anim01;  
  92.     if (batteryHealth >= 10 && batteryHealth < 15)  
  93.       return R.drawable.stat_sys_battery_charge_anim02;  
  94.     if (batteryHealth >= 15 && batteryHealth < 20)  
  95.       return R.drawable.stat_sys_battery_charge_anim03;  
  96.     if (batteryHealth >= 20 && batteryHealth < 25)  
  97.       return R.drawable.stat_sys_battery_charge_anim04;  
  98.     if (batteryHealth >= 25 && batteryHealth < 30)  
  99.       return R.drawable.stat_sys_battery_charge_anim05;  
  100.     if (batteryHealth >= 30 && batteryHealth < 35)  
  101.       return R.drawable.stat_sys_battery_charge_anim06;  
  102.     if (batteryHealth >= 35 && batteryHealth < 40)  
  103.       return R.drawable.stat_sys_battery_charge_anim07;  
  104.     if (batteryHealth >= 40 && batteryHealth < 45)  
  105.       return R.drawable.stat_sys_battery_charge_anim08;  
  106.     if (batteryHealth >= 45 && batteryHealth < 50)  
  107.       return R.drawable.stat_sys_battery_charge_anim09;  
  108.     if (batteryHealth >= 50 && batteryHealth < 55)  
  109.       return R.drawable.stat_sys_battery_charge_anim10;  
  110.     if (batteryHealth >= 55 && batteryHealth < 60)  
  111.       return R.drawable.stat_sys_battery_charge_anim11;  
  112.     if (batteryHealth >= 60 && batteryHealth < 65)  
  113.       return R.drawable.stat_sys_battery_charge_anim12;  
  114.     if (batteryHealth >= 65 && batteryHealth < 70)  
  115.       return R.drawable.stat_sys_battery_charge_anim13;  
  116.     if (batteryHealth >= 70 && batteryHealth < 75)  
  117.       return R.drawable.stat_sys_battery_charge_anim14;  
  118.     if (batteryHealth >= 75 && batteryHealth < 80)  
  119.       return R.drawable.stat_sys_battery_charge_anim15;  
  120.     if (batteryHealth >= 80 && batteryHealth < 85)  
  121.       return R.drawable.stat_sys_battery_charge_anim16;  
  122.     if (batteryHealth >= 85 && batteryHealth < 90)  
  123.       return R.drawable.stat_sys_battery_charge_anim17;  
  124.     if (batteryHealth >= 90 && batteryHealth < 95)  
  125.       return R.drawable.stat_sys_battery_charge_anim18;  
  126.     if (batteryHealth >= 95 && batteryHealth < 100)  
  127.       return R.drawable.stat_sys_battery_charge_anim19;  
  128.     if (batteryHealth == 100)  
  129.       return R.drawable.stat_sys_battery_charge_animfull;  
  130.     return R.drawable.stat_sys_battery_unknown;  
  131.   }  
  132.  
  133.   // -------------------------------私有字段--------------------------------------  
  134.   private NotificationManager notifyManager = null;  
  135.   private Notification notification = null;  
  136.  
  137.   //这里是为了方便起见而硬编码的icon资源数组,直接从R.java文件中提取的十六进制资源ID,本数组的0号元素0x7f020002表示资源stat_sys_battery_0.png(见附件),以后累加。不同配置的ID可能不同,请读者自行修改。  
  138.   private int batteryStateIcons[] = { 0x7f0200020x7f0200030x7f020004,  
  139.       0x7f0200050x7f0200060x7f0200070x7f0200080x7f0200090x7f02000a,  
  140.       0x7f02000b0x7f02000c0x7f02000d0x7f02000e0x7f02000f0x7f020010,  
  141.       0x7f0200110x7f0200120x7f0200130x7f0200140x7f0200150x7f020016,  
  142.       0x7f0200170x7f0200180x7f0200190x7f02001a0x7f02001b0x7f02001c,  
  143.       0x7f02001d0x7f02001e0x7f02001f0x7f0200200x7f0200210x7f020022,  
  144.       0x7f0200230x7f0200240x7f0200250x7f0200260x7f0200270x7f020028,  
  145.       0x7f0200290x7f02002a0x7f02002b0x7f02002c0x7f02002d0x7f02002e,  
  146.       0x7f02002f0x7f0200300x7f0200310x7f0200320x7f0200330x7f020034,  
  147.       0x7f0200350x7f0200360x7f0200370x7f0200380x7f0200390x7f02003a,  
  148.       0x7f02003b0x7f02003c0x7f02003d0x7f02003e0x7f02003f0x7f020040,  
  149.       0x7f0200410x7f0200420x7f0200430x7f0200440x7f0200450x7f020046,  
  150.       0x7f0200470x7f0200480x7f0200490x7f02004a0x7f02004b0x7f02004c,  
  151.       0x7f02004d0x7f02004e0x7f02004f0x7f0200500x7f0200510x7f020052,  
  152.       0x7f0200530x7f0200540x7f0200550x7f0200560x7f0200570x7f020058,  
  153.       0x7f0200590x7f02005a0x7f02005b0x7f02005c0x7f02005d0x7f02005e,  
  154.       0x7f02005f0x7f0200600x7f0200610x7f0200620x7f0200630x7f020064,  
  155.       0x7f0200650x7f02007b };  
  156.  
  157.   // 接受电池信息更新的广播  
  158.   private BroadcastReceiver batteryChangedReceiver = new BroadcastReceiver() {  
  159.     public void onReceive(Context context, Intent intent) {  
  160.       int level = intent.getIntExtra("level"0);  
  161.       int scale = intent.getIntExtra("scale"100);  
  162.       int status = intent.getIntExtra("status"0);  
  163.  
  164.       // 若正在充电  
  165.       if (status == BatteryManager.BATTERY_STATUS_CHARGING)  
  166.         notification = getNotification(getChargingIcon(level * 100 / scale),  
  167.             "Battery Monitor", System.currentTimeMillis(), "电池电量""正在充电");  
  168.       else 
  169.         notification = getNotification(getStateIcon(level * 100 / scale),  
  170.             "Battery Monitor", System.currentTimeMillis(), "电池电量""请及时充电");  
  171.       notifyManager.notify(0, notification);  
  172.     }  
  173.   };  

光有心,不跳怎么行?——BatteryMonitor

通过上面的服务,我们就能得到一个在左上角显示当前电量的通知图标。现在心脏倒是被我们大家轻而易举地造出来了,下一步当然是让它跳动起来!

首先,我们需要将MonitorService服务在系统配置文件中进行注册。

 


  
  1. <!-- service --> 
  2.  <service android:name=".MonitorService" android:enabled="true" /> 

然后,我们需要一个心脏起搏器——Activity(或者直接把本服务作为开机启动项,请见Bill的另一篇文章【Android开机启动ActivityService的方法】),下面,我们将在一个Activity中启动MonitorService服务。

    首先我们需要将这个ActivityAndroidManifest.xml中进行注册

 


  
  1. <!-- activity --> 
  2. <activity android:name=".BatteryMonitor" android:label="@string/battery_monitor_name"> 
  3.     <intent-filter> 
  4.     <action android:name="android.intent.action.MAIN" />   
  5.     <category android:name="android.intent.category.LAUNCHER" /> 
  6.     </intent-filter> 
  7. </activity> 

    然后,在BatteryMonitor.java中编码启动我们的MonitorService服务即可

 


  
  1. package com.billhoo.study;  
  2.  
  3. import android.app.Activity;  
  4. import android.content.Intent;  
  5. import android.os.Bundle;  
  6.  
  7.  
  8. public class BatteryMonitor extends Activity {  
  9.   /** Called when the activity is first created. */ 
  10.   @Override 
  11.   public void onCreate(Bundle savedInstanceState) {  
  12.     super.onCreate(savedInstanceState);  
  13.     setContentView(R.layout.main);  
  14.       
  15.     //启动电池监控服务  
  16.     startService(new Intent(this, MonitorService.class));  
  17.   }  

 

至此,我们的电池监控小程序已经初见成效,大家可以启动我们的Activity,看看左上角是不是出现了一个小电池图标,并显示着当前电量?

好了,下面来让我们的程序邪恶滴向用户的桌面进军吧!

 

麻雀虽小,五脏俱全——MonitorWidgetProvider为麻雀穿上衣服

众所周知,Android用户会将自己喜爱的小程序放在桌面以便快速启动,当然我们就不能放过这个机会!

但是,仅凭我们的Activity还做不到这一点,难道我们应该期望用户在他的程序列表中长按我们的Activity,然后使出吃奶的劲儿把它扔在桌面上吗?不,他们通常长按桌面,然后轻点一个自己中意的Widget插件便能满足他们的需求。

因此,我们的监控程序也应该有自己的Widget,一来方便用户查看电池状态(仅凭左上角的通知有时候并不能满足用户的需求),二来方便用户随时启动本程序附带的电池详细数据报告(本功能并未在本文中实现)。

下面就让我们一起来实现这个Widget!

我们知道,Widget本身就是一个广播接收器(BroadcastReceiver),因此我们最初企图直接用这个Widget接受ACTION_BATTERY_CHANGED广播并作出反应,但却发现这个接收器并不会接受电池更新的广播(不论我们在配置文件中配置与否)。所以只能另寻出路,我们想到,可以使用一个单独的为本Widget服务的Service,该Service检测到电池电量更新后就更新桌面的Widget显示。于是便有以下Widget及其配套Service的实现。

首先,我们需要为自己的Widget创建一个配置文件,用于描述本Widget在桌面显示时是个什么样子。

 


  
  1. <!-- 项目路径\res\layout\monitor_widget_layout.xml -->   
  2. <?xml version="1.0" encoding="utf-8"?> 
  3. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"   
  4.     android:orientation="vertical" 
  5.     android:layout_width="72dip" 
  6.     android:layout_height="72dip"> 
  7.     <ImageView android:id="@+id/monitor_widget_view" 
  8.      android:layout_width="fill_parent" 
  9.     android:layout_height="fill_parent"
  10.     android:src="@drawable/vertical_battery_unknown" /> 
  11.  </LinearLayout> 

 


  
  1. <!-- 项目路径\res\xml\monitor_widget_info.xml (xml文件夹是自行创建的)--> 
  2. <?xml version="1.0" encoding="utf-8"?> 
  3. <appwidget-providerxmlns:androidappwidget-providerxmlns:android="http://schemas.android.com/apk/res/android"               android:initialLayout="@layout/monitor_widget_layout" 
  4. android:minWidth="72dip" android:minHeight="72dip" /> 

    然后我们需要在AndroidManifest.xml配置文件中注册我们的Widget

 


  
  1. <receiver android:name=".MonitorWidgetProvider">
  2. <intent-filter> 
  3. <action android:name="android.appwidget.action.APPWIDGET_UPDATE" /> 
  4. <action android:name="android.intent.action.BATTERY_CHANGED" /> 
  5. </intent-filter> 
  6. <meta-data android:name="android.appwidget.provider" 
  7.   android:resource="@xml/monitor_widget_info" /> 
  8. </receiver> 

最后,我们需要编写本Widget的实现代码

 


  
  1. package com.billhoo.study;  
  2.  
  3. import android.appwidget.AppWidgetManager;  
  4. import android.appwidget.AppWidgetProvider;  
  5. import android.content.Context;  
  6. import android.content.Intent;  
  7.  
  8. public class MonitorWidgetProvider extends AppWidgetProvider   
  9. {  
  10.   //当Widget第一次被启动时,启动配套服务  
  11.   @Override 
  12.   public void onEnabled(Context context) {  
  13.     context.startService(new Intent(context,   
  14.  
  15. WidgetUpdateService.class));  
  16.   }  
  17.  
  18.   //当Widget更新时,启动配套服务  
  19.   @Override 
  20.   public void onUpdate(Context context, AppWidgetManager   
  21.  
  22. appWidgetManager,  
  23.       int appWidgetIds[]) {  
  24.     super.onUpdate(context, appWidgetManager, appWidgetIds);  
  25.     context.startService(new Intent(context,   
  26.  
  27. WidgetUpdateService.class));  
  28.   }  
  29.  
  30.   // 当桌面上这个widget的所有实体都被删除后,停止配套服务  
  31.   @Override 
  32.   public void onDisabled(Context context) {  
  33.     context.stopService(new Intent(context, WidgetUpdateService.class));  
  34.   }  

    下面让我们来为这个Widget编写配套的Service。

    首先在AndroidManifest.xml中注册这个服务。

 


  
  1. <service android:name=".WidgetUpdateService" android:enabled="true" /> 

    然后,编写其JAVA实现代码

 


  
  1. package com.billhoo.study;  
  2.  
  3. import android.app.Service;  
  4. import android.appwidget.AppWidgetManager;  
  5. import android.content.BroadcastReceiver;  
  6. import android.content.ComponentName;  
  7. import android.content.Context;  
  8. import android.content.Intent;  
  9. import android.content.IntentFilter;  
  10. import android.os.BatteryManager;  
  11. import android.os.IBinder;  
  12. import android.widget.RemoteViews;  
  13.  
  14. //用以根据电池电量更新而刷新桌面Widget的服务  
  15. public class WidgetUpdateService extends Service {  
  16.   @Override 
  17.   public IBinder onBind(Intent intent) {  
  18.     return null;  
  19.   }  
  20.  
  21.   @Override 
  22.   public void onStart(Intent intent, int startId) {  
  23.     // 向系统注册电池电量更新广播,只接受带有ACTION_BATTERRY_CHANGED事件的Intent  
  24.     registerReceiver(this.bCR, new IntentFilter(Intent.ACTION_BATTERY_CHANGED));  
  25.  
  26.     bCR = new BroadcastReceiver() {  
  27.       @Override 
  28.       public void onReceive(Context context, Intent intent) {  
  29.         int level = intent.getIntExtra("level"0);  
  30.         int scale = intent.getIntExtra("scale"100);  
  31.         int status = intent.getIntExtra("status"0);  
  32.         int batteryHealth = level * 100 / scale;  
  33.         int icon = 0;  
  34.  
  35.         AppWidgetManager widgetManager = AppWidgetManager.getInstance(context);  
  36.         RemoteViews widgetView = new RemoteViews(context.getPackageName(),  
  37.             R.layout.monitor_widget_layout);  
  38.  
  39.         // 若正在充电  
  40.         if (status == BatteryManager.BATTERY_STATUS_CHARGING) {  
  41.           icon = getChargingIcon(batteryHealth);  
  42.           widgetView.setImageViewResource(R.id.monitor_widget_view, icon  
  43.  
  44.           );  
  45.         } else 
  46.           widgetView.setImageViewResource(R.id.monitor_widget_view,  
  47.               getStateIcon(batteryHealth));  
  48.         widgetManager.updateAppWidget(new ComponentName(context,  
  49.             MonitorWidgetProvider.class), widgetView);  
  50.       }  
  51.     };  
  52.   }  
  53.  
  54.   private BroadcastReceiver bCR = null;  
  55.  
  56.   // 获取电池状态图标  
  57.   private int getStateIcon(int batteryHealth) {  
  58.     if (batteryHealth >= 0 || batteryHealth <= 100)  
  59.       return batteryStateIcon[batteryHealth / 5];  
  60.     return R.drawable.vertical_battery_unknown;  
  61.   }  
  62.  
  63.   // 获得对应的充电状态图标  
  64.   private int getChargingIcon(int batteryHealth) {  
  65.     if (batteryHealth >= 0 && batteryHealth < 5)  
  66.       return R.drawable.vertical_charging_00;  
  67.     if (batteryHealth >= 5 && batteryHealth < 10)  
  68.       return R.drawable.vertical_charging_01;  
  69.     if (batteryHealth >= 10 && batteryHealth < 15)  
  70.       return R.drawable.vertical_charging_02;  
  71.     if (batteryHealth >= 15 && batteryHealth < 20)  
  72.       return R.drawable.vertical_charging_03;  
  73.     if (batteryHealth >= 20 && batteryHealth < 25)  
  74.       return R.drawable.vertical_charging_04;  
  75.     if (batteryHealth >= 25 && batteryHealth < 30)  
  76.       return R.drawable.vertical_charging_05;  
  77.     if (batteryHealth >= 30 && batteryHealth < 35)  
  78.       return R.drawable.vertical_charging_06;  
  79.     if (batteryHealth >= 35 && batteryHealth < 40)  
  80.       return R.drawable.vertical_charging_07;  
  81.     if (batteryHealth >= 40 && batteryHealth < 45)  
  82.       return R.drawable.vertical_charging_08;  
  83.     if (batteryHealth >= 45 && batteryHealth < 50)  
  84.       return R.drawable.vertical_charging_09;  
  85.     if (batteryHealth >= 50 && batteryHealth < 55)  
  86.       return R.drawable.vertical_charging_10;  
  87.     if (batteryHealth >= 55 && batteryHealth < 60)  
  88.       return R.drawable.vertical_charging_11;  
  89.     if (batteryHealth >= 60 && batteryHealth < 65)  
  90.       return R.drawable.vertical_charging_12;  
  91.     if (batteryHealth >= 65 && batteryHealth < 70)  
  92.       return R.drawable.vertical_charging_13;  
  93.     if (batteryHealth >= 70 && batteryHealth < 75)  
  94.       return R.drawable.vertical_charging_14;  
  95.     if (batteryHealth >= 75 && batteryHealth < 80)  
  96.       return R.drawable.vertical_charging_15;  
  97.     if (batteryHealth >= 80 && batteryHealth < 85)  
  98.       return R.drawable.vertical_charging_16;  
  99.     if (batteryHealth >= 85 && batteryHealth < 90)  
  100.       return R.drawable.vertical_charging_17;  
  101.     if (batteryHealth >= 90 && batteryHealth < 95)  
  102.       return R.drawable.vertical_charging_18;  
  103.     if (batteryHealth >= 95 && batteryHealth < 100)  
  104.       return R.drawable.vertical_charging_19;  
  105.     if (batteryHealth == 100)  
  106.       return R.drawable.vertical_battery_full;  
  107.     return R.drawable.stat_sys_battery_unknown;  
  108.   }  
  109.  
  110.   private int batteryStateIcon[] = { 0x7f02007d0x7f02007e0x7f02007f,  
  111.       0x7f0200800x7f0200810x7f0200820x7f0200830x7f0200840x7f020085,  
  112.       0x7f0200860x7f0200870x7f0200880x7f0200890x7f02008a0x7f02008b,  
  113.       0x7f02008c0x7f02008d0x7f02008e0x7f02008f0x7f0200900x7f020091 };  

至此,我们的电池电量监控程序也算告一段落,这样,我们就不需要在忍受android自带的电池图标了,何乐不为?

 

让麻雀自食其力——开机启动

不过,我们的小麻雀的确还有点天然呆!需要你主动去找它,而不是它主动为你服务!这个不是好事情,尤其是对于电池电量监控来说。因此,我们需要对它做最后的改造——开机启动。关于开机启动的详细步骤bill已经在【Android开机启动ActivityService的方法】一文中详述,故恕我不再赘述。

 

写在最后

这只是一个非常粗糙的练手程序,我们可以参照网络上各种新颖实用的Android电池监控程序来不断完善自己的小玩意儿,在完善的过程中,就是一次对自己的历练。

最后希望大家在新的一年里都有新的收获,新的人生精彩!

 

 

 

转载于:https://blog.51cto.com/billhoo/762776

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【Android电池电量监控初步】 的相关文章

  • IdentityServer4 使用OpenID Connect添加用户身份验证

    使用IdentityServer4 实现OpenID Connect服务端 xff0c 添加用户身份验证 客户端调用 xff0c 实现授权 IdentityServer4 目前已更新至1 0 版 xff0c 在之前的文章中有所介绍 Iden
  • ftp服务器覆盖文件,ftp 覆盖 服务器 文件

    ftp 覆盖 服务器 文件 内容精选 换一换 本章节适用于MRS 3 x之前版本 Loader支持以下多种连接 xff0c 每种连接的配置介绍可根据本章节内容了解 obs connectorgeneric jdbc connectorftp
  • Ubuntu-14.04. sh .py腳本双击無法執行问题的解决方法

    Ubuntu 14 04中默认文件用gedit文本打开 xff0c 而不是BT5里面的默认双击打开四个选择 xff0c 例如以下图 xff08 这是配置完毕后的结果 xff0c 就不换BT5系统了 xff09 xff1a 直接文本打开 xf
  • Ubuntu解决循环登陆问题

    问题描述 xff1a 安装驱动之后重启 xff0c 一直循环在登录页面 解决方法 xff1a Ctrl 43 Alt 43 F1 xff0c 进入命令行界面 1 卸载原来的驱动 sudo apt get remove purge nvidi
  • 【Linux/CentOS】Boolean ftp_home_dir is not defined

    安装完vsftpd软件后 xff0c 因为CentOS系统的SELinux安全策略默认是没有开启FTP服务 xff0c 直接访问会报错500 OOPS xff0c 所以需要修改为允许使用FTP服务 目标 xff1a 希望ftp用户可以访问自
  • LINUX中 设置登录超时

    ssh设置超时时间 ssh连接超时问题解决方案 xff1a 1 修改server端的etc ssh sshd config Client AliveInterval 60 xff03 server每隔60秒发送一次请求给client xff
  • eclipse 停止java程序运行_Eclipse:停止运行代码(java)

    有时 xff0c 我会运行一个偶然包含无限循环之类的程序 Eclipse让我继续编辑程序 xff0c 但速度非常慢 我该如何阻止它 xff1f 我是否想重新启动JVM xff1f 重新启动eclipse本身总是可行的 xff0c 但这会中断
  • Linux 的五个重启命令及具体说明

    一 Linux 的五个重启命令 1 shutdown 2 poweroff 3 init 4 reboot 5 halt 二 五个重启命令的具体说明 shutdown reboot 在linux下一些常用的关机 重启指令有shutdown
  • 如何直接在ftp里编辑文件

    首先要连接ftp xff0c 如何链接ftp 这个我已经分享过 xff0c 如果还有不懂的 xff0c 可以查看下经验分享 xff0c 打开ftp xff0c 并连接你要修改的站点 xff01 下载安装代码编辑器 xff0c 比如EditP
  • 什么是真正的程序员?

    什么是真正的程序员 这篇文章的原文来自 xff1a A Little Printf Story 作者仿照 小王子 中的情节 xff0c 通过小printf遇见的不同类型的程序员 xff0c 最后悟出什么才是真正的程序员 xff01 第一次翻
  • python import ssl失败_Python3.7.5 ssl模块导入失败的解决办法

    import ssl if we can 39 t import it let the error propagate ImportError DLL load failed 找不到指定的程序 遇到问题的过程 xff0c 在windows7
  • Linux学习之CentOS(三十)--SELinux安全系统基础

    您可以通过点击 右下角 的按钮 来对文章内容作出评价 也可以通过左下方的 关注按钮 来关注我的博客的最新动态 如果文章内容对您有帮助 不要忘记点击右下角的 推荐按钮 来支持一下哦 如果您对文章内容有任何疑问 可以通过评论或发邮件的方式联系我
  • PHP中__FUNCTION__与__METHOD__的区别

    你知道php中 FUNCTION 与 METHOD 的区别吗 xff1f 本文通过一个小例子 xff0c 为大家介绍下二者的区别 xff0c 有兴趣的朋友可以参考下 PHP中 FUNCTION 与 METHOD 的区别 xff0c 主要在以
  • linux设置登录超时,登录失败策略

    1 登录超时 在vi etc profile 末尾加上 如果有就修改 没有就添加 export TMOUT 61 600 单位为秒 10分钟 2 登录失败 在 etc pam d system auth 加上 deny 61 3错误次数 u
  • SecureCRT分屏显示

    Tab右键 或者 Session Manager右键 gt Send to New Tab Group
  • WPF DataGrid多表头/列头,多行头,合并单元格,一列占据多行

    原文 WPF DataGrid多表头 列头 xff0c 多行头 xff0c 合并单元格 xff0c 一列占据多行 先上效果图 xff1a 思路说明 xff1a 这是两个DataGrid xff0c 没有嵌套 xff0c 位置和高度保持一致
  • cookie和session同与异

    面试和工作中有时候碰到cookie和session的同异 xff0c 现在小结如下 xff1a 首先看cookie cookie增删改查 xff1a http www w3school com cn js 1 cookie产生 识别用户 用
  • 阿里云Redis配置公网连接的注意点

    本篇文章是对官方文档中关于配置redis外网链接的补充 xff0c 请大家参考 Redis公网连接 前提条件 要保证进行外网链接配置的这个ecs是能够连接上redis实例的 最简单的办法是在ecs上能够使用ping命令进行验证 xff0c
  • echarts图例legend选中状态动态设置 selected

    功能前端页面展示 xff0c 需要对所有指标的图例默认选中状态只显示前两个 xff0c 其他指标的图例状态默认为灰色 1 因为所有的指标为动态加载的 xff0c 因此图例legend选中状态也要通过js进行动态加载 xff0c 从官方的文档
  • 如何给run()方法传参数

    实现的方式主要有三种 1 构造函数传参 2 成员变量传参 3 回调函数传参 问题 xff1a 如何实现处理线程的返回值 xff1f 1 主线程等待法 xff08 优点 xff1a 实现起来简单 xff0c 缺点 xff1a 需要等待的变量一

随机推荐

  • devC++代码格式化对齐的快捷键

    devC 43 43 代码格式化对齐的快捷键是ctrl 43 shift 43 a ctrl 43 左右键可以使光标移动一个单词的距离 shirt 43 左右键可以选中光标左右的一个字符 转载于 https www cnblogs com
  • PostgreSQL 使用PG_Rman进行物理备份

    背景 在Oracle下我们可以使用rman进行物理备份 xff0c 支持数据库的全量 增量 归档的备份模式 而PostgreSQL作为开源数据库 xff0c 近些时间来也一直向商业版数据库看齐 xff0c 也推出了开源功工具pg rman
  • 引用计数的智能指针的实现与思考

    摘要 引用计数在软件开发中是一项非常重用的技术 xff0c 它可以说是无处不 xff0c 我们在不知不觉中都在和它打交道 xff0c 比如 Windows上的COM和Handle xff0c Mac上的ref句柄 xff0c 脚本语言中的垃
  • test

    1 overrides the s4 notdlg class items as display none lt script type 61 34 text javascript 34 gt var fV4UI 61 true lt sc
  • keil5 --工程创建

    一 xff0c 文件夹介绍 首先去官网过去其他地方获取到官方提供的标准库文件 下面这个我是在官网进行下载的 我们在打开keil的时候会弹出一个在线下载的框 xff08 这个框这里先不做说明 xff0c 后面在继续讲解 xff09 xff0c
  • gnome-tweak-tool设置gnome参数, 修改CENTOS7桌面图标大小

    GNOME Tweak Tool 是 GNOME 3 的优化配置工具 xff0c 为我们带来 GNOME Shell 扩展安装功能 xff0c 方便Linux用户对 Gnome Shell 进行一些调整 主要功能有 xff1a 安装 xff
  • linux判断usb进程命令,一种在Linux系统下审计USB设备历史使用情况的方法与流程...

    本发明涉及计算机审计技术领域 xff0c 具体涉及一种在Linux系统下审计USB设备历史使用情况的方法 背景技术 xff1a 如今 xff0c 在linux系统中 xff0c 对于USB设备的插入拔出事件 xff0c 系统自身是不带有审计
  • 又是一年年终总结

    起 这篇年终总结草稿是在12 03起的 xff0c 那是突然之间感觉到今年不大平常 xff0c 可以考虑写个年终总结来记录一下 xff0c 但是谁能料到今年真的是太不平常了 xff0c 到了12月中 xff0c 公司就解散了 xff0c 所
  • 远程连接windows系统提示:其他用户要远程登录,需要通过远程桌面服务进行登录的权限......

    解决方法 xff1a 服务器内部 通过在 本地组策略编辑器 中 计算机配置 gt Windows设置 gt 安全设置 gt 本地策略 gt 用户权限分配 进行相关调试即可 删除即可 转载于 https blog 51cto com 1377
  • 【封装那些事】 缺失封装

    缺失封装 没有将实现变化封装在抽象和层次结构中时 xff0c 将导致这种坏味 表现形式通常如下 客户程序与其需要的服务变种紧密耦合 xff0c 每当需要支持新变种或修改既有变种时 xff0c 都将影响客户程序 每当需要在层次结构中支持新变种
  • RxJava 和 RxAndroid 五(线程调度)

    对rxJava不了解的同学可以先看 RxJava 和 RxAndroid 一 基础 RxJava 和 RxAndroid 二 xff08 操作符的使用 xff09 RxJava 和 RxAndroid 三 xff08 生命周期控制和内存优化
  • PHP版本切换

    前言 php是为了快速构建一个web页面而迅速被大家广为接受的开源语言 xff0c 通过不断发展已经有了很多的php开源系统 xff0c 满足了目前大部分用户的站点需求 1995年初php诞生到现在已经存在多个版本 xff0c 并且每个版本
  • 成功不是依靠机会 ---- 参加移动开发者大会大会有感

    这次有幸参加了CSDN和创新工厂主办的移动开发者大会 xff0c 感觉良多 第一印象是 xff1a 这真的是一次技术的大会 我之前参加过很多大会 我特别说的是微软的技术大会 xff0c 已经感受不到什么技术的味道了 xff0c 或者说是这种
  • CentOS 7命令行安装图形界面(GNOME、KDE等)

    CentOS 7 默认是没有图形化界面的 xff0c 但我们很多人在习惯了 Windows 的图形化界面之后 xff0c 总是希望有一个图形化界面从而方便我们使用 xff0c 这里介绍一下 CentOS xff17 安装图形化桌面系统的方法
  • 我的世界服务器皮肤怎么用文件夹,我的世界怎么用皮肤文件,怎么通过文件夹更改皮肤...

    打开versions xff0c 我的世界怎么用文件换皮肤教程里百面有个小茶壶形状的文件 xff0c 用压度缩工具打开它 xff0c 依次打知开assets xff0c minecraft xff0c xff0c textures xff0
  • firewald、netfilter、iptables介绍及表案例

    1 firewalld和netfilter 2 netfilter5表5链介绍 3 iptables语法 4 iptables filter表案例 5 iptables nat表应用 1 firewalld和netfilter centos
  • 一个老兵的linux学习和面试经验分享

    特别说明 xff1a 本文为约9个月前老男孩linux培训内部师兄给师弟的经验分享 xff0c 经过该同学同意 xff0c 特此分享给所有博友 学习和面试经验分享 大家好 xff0c 非常高兴能在这里给大家分享学习和面试的经验 xff0c
  • Android原生控件介绍

    安卓开发终极指南 50 多个初高级开发资源 xff08 译 xff09 我仍记得几年前刚开始进入 Android 开发这个广阔而又神秘的世界时 xff0c 手足无措的样子 为了帮助像我这样的开发者 xff0c 我整理了一份比较全的学习资料
  • docker login 登录_Docker login 命令-Docker login 作用-Docker login 默认用户名和密码-Docker登录-嗨客网...

    docker login命令教程 docker login 命令用于登陆到一个 Docker 镜像仓库 xff0c 如果未指定镜像仓库地址 xff0c 默认为官方仓库 Docker Hub 如果用户使用 docker login 命令登录官
  • 【Android电池电量监控初步】

    引言 初学Andorid xff0c 学习了包括Activity Service AppWidget Broadcast Intent Notification XML布局 XML配置等基础知识之后 xff0c 尝试着做一个小玩意儿 电池监