Android闹钟设置的解决方案

简介:

Android设置闹钟并不像IOS那样这么简单,做过Android设置闹钟的开发者都知道里面的坑有多深。下面记录一下,我解决Android闹钟设置的解决方案。

主要问题

  1. API19开始AlarmManager的机制修改。
  2. 应用程序被Kill掉后,设置的闹钟不响。
  3. 6.0以上进入Doze模式会使JobScheduler停止工作。
  4. 手机设置重启后,闹钟失效问题。

API19以上AlarmManager机制的修改

API19之前AlarmManager提供了三个设置闹钟的方法,由于业务需求闹钟只需要一次性,所以采用set(int type,long startTime,PendingIntent pi);这个方法。从API 19开始,AlarmManager的机制都是非准确传递,操作系统将会转换闹钟,来最小化唤醒和电池使用。

由于之前的程序,没有对API19以上的闹钟设置做处理,导致在4.4以上的手机设置闹钟无响应(应用程序没有被杀死的情况也没有闹钟)。

因些,设置闹钟需要根据API的版本进行分别处理设置。代码如下:

 
 
  1. AlarmManager am = (AlarmManager) getActivity() 
  2.        .getSystemService(Context.ALARM_SERVICE); 
  3. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { 
  4.     am.setExact(AlarmManager.RTC_WAKEUP, TimeUtils 
  5.         .stringToLong(recordTime, TimeUtils.NO_SECOND_FORMAT), sender); 
  6. }else { 
  7.     am.set(AlarmManager.RTC_WAKEUP, TimeUtils 
  8.         .stringToLong(recordTime, TimeUtils.NO_SECOND_FORMAT), sender); 

这样,保证闹钟在应用程序没有被Kill掉的情况闹钟。

应用程序被Kill掉时的处理

应用程序被Kill掉后,设置的闹钟失效,这里利用守护进程以及灰色保活来保证后台闹钟服务不被Kill掉。当应用程序以及闹钟服务被Kill掉,守护进程以及灰色保活来重新启动闹钟服务,并且重新设置闹钟。

关于守护进程的处理,这里采用开源的守护进程库。Android-AppDaemon

在闹钟服务的onCreat加入Android-AppDaemon这个开源的守护进程。代码如下:

 
 
  1. @Override 
  2. public void onCreate() { 
  3.     super.onCreate(); 
  4.     Daemon.run(DaemonService.this,  
  5.            DaemonService.class, Daemon.INTERVAL_ONE_MINUTE); 
  6.     startTimeTask(); 
  7.     grayGuard(); 

为进一步保证闹钟服务的存活,同加上灰色保活(利用系统的漏洞启动前台Service)。代码如下:

 
 
  1. private void grayGuard() { 
  2.     if (Build.VERSION.SDK_INT < 18) { 
  3.         //API < 18 ,此方法能有效隐藏Notification上的图标 
  4.         startForeground(GRAY_SERVICE_ID, new Notification()); 
  5.     } else { 
  6.         Intent innerIntent = new Intent(this, DaemonInnerService.class); 
  7.         startService(innerIntent); 
  8.         startForeground(GRAY_SERVICE_ID, new Notification()); 
  9.     } 
  10.  
  11.     //发送唤醒广播来促使挂掉的UI进程重新启动起来 
  12.     AlarmManager alarmManager = (AlarmManager) getSystemService(Context.ALARM_SERVICE); 
  13.     Intent alarmIntent = new Intent(); 
  14.     alarmIntent.setAction(WakeReceiver.GRAY_WAKE_ACTION); 
  15.     PendingIntent operation = PendingIntent.getBroadcast(this,  
  16.         WAKE_REQUEST_CODE, alarmIntent, PendingIntent.FLAG_UPDATE_CURRENT); 
  17.     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { 
  18.         alarmManager.setWindow(AlarmManager.RTC_WAKEUP,  
  19.             System.currentTimeMillis(), ALARM_INTERVAL, operation); 
  20.     }else { 
  21.         alarmManager.setInexactRepeating(AlarmManager.RTC_WAKEUP,  
  22.             System.currentTimeMillis(), ALARM_INTERVAL, operation); 
  23.     } 
  24.  
  25. /** 
  26.  * 给 API >= 18 的平台上用的灰色保活手段 
  27.  */ 
  28. public static class DaemonInnerService extends Service { 
  29.  
  30.     @Override 
  31.     public void onCreate() { 
  32.         Log.i(LOG_TAG, "InnerService -> onCreate"); 
  33.         super.onCreate(); 
  34.     } 
  35.  
  36.     @Override 
  37.     public int onStartCommand(Intent intent, int flags, int startId) { 
  38.         Log.i(LOG_TAG, "InnerService -> onStartCommand"); 
  39.         startForeground(GRAY_SERVICE_ID, new Notification()); 
  40.         //stopForeground(true); 
  41.         stopSelf(); 
  42.         return super.onStartCommand(intent, flags, startId); 
  43.     } 
  44.  
  45.     @Override 
  46.     public IBinder onBind(Intent intent) { 
  47.         throw new UnsupportedOperationException("Not yet implemented"); 
  48.     } 
  49.  
  50.     @Override 
  51.     public void onDestroy() { 
  52.         Log.i(LOG_TAG, "InnerService -> onDestroy"); 
  53.         super.onDestroy(); 
  54.     } 

上面操作尽可能提高闹钟服务的存活。但是在5.0以上的手机,利用系统的自带的Clean功能的时候,还是会将闹钟服务彻底的干掉。为了解决5.0以上的问题,这里引入5.0以上的新特性 JobScheduler。

5.0以上的JobScheduler

关于5.0新增JobScheduler·API可以先阅读这篇文章。here

在这里利用5.0以上的JobScheduler创建一个定时的任务,定时检测闹钟服务是否存在,没在存在则重新启动闹钟服务。(这里我设置每一分钟检测一次闹钟服务)

在进入应用程序的时候检测当前系统是否是5.0以上,如果是则启动JobScheduler这个服务。代码如下:

 
 
  1. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { 
  2.     mJobScheduler = (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE); 
  3.     JobInfo.Builder builder = new JobInfo.Builder(JOB_ID, 
  4.             new ComponentName(getPackageName(), JobSchedulerService.class.getName())); 
  5.  
  6.     builder.setPeriodic(60 * 1000); //每隔60秒运行一次 
  7.     builder.setRequiresCharging(true); 
  8.     builder.setPersisted(true);  //设置设备重启后,是否重新执行任务 
  9.     builder.setRequiresDeviceIdle(true); 
  10.  
  11.     if (mJobScheduler.schedule(builder.build()) <= 0) { 
  12.         //If something goes wrong 
  13.     } 
  14.  

其中的builder.setPersisted(true); 方法是设备重启后,是否重新执行任务,在这测过是可以重新启动任务的。

上面的操作进一步保证了闹钟服务被Kill掉后,重新启动服务。但是在6.0以上引入了Doze模式,当6.0以上的手机进入这个模式后,便会使JobScheduler停止工作。

6.0以上Doze模式的处理

为了让JobScheduler可以在6.0以上进入Doze模式工作,这里针对6.0以上的Doze模式做特殊的处理-忽略电池的优化。

  • 在Manifest.xml中加入权限。

<uses-permission android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS"/>

  • 在设置闹钟的时候,判断系统是否是6.0以上,如果是,则判断是否忽略电池的优化。判断是否忽略电池优化代码如下:
 
 
  1. TargetApi(Build.VERSION_CODES.M) 
  2. public static boolean isIgnoringBatteryOptimizations(Activity activity){ 
  3.     String packageName = activity.getPackageName(); 
  4.     PowerManager pm = (PowerManager) activity 
  5.             .getSystemService(Context.POWER_SERVICE); 
  6.     if (pm.isIgnoringBatteryOptimizations(packageName)) { 
  7.         return true
  8.     }else { 
  9.         return false
  10.     } 
  11.  

如果没有忽略电池优化的时候,弹出提醒对话框,提示用户进行忽略电池优化操作。代码如下:

 
 
  1. /** 
  2.  * 针对N以上的Doze模式 
  3.  * 
  4.  * @param activity 
  5.  */ 
  6. public static void isIgnoreBatteryOption(Activity activity) { 
  7.     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { 
  8.         try { 
  9.             Intent intent = new Intent(); 
  10.             String packageName = activity.getPackageName(); 
  11.             PowerManager pm = (PowerManager) activity.getSystemService(Context.POWER_SERVICE); 
  12.             if (!pm.isIgnoringBatteryOptimizations(packageName)) { 
  13. //               intent.setAction(Settings.ACTION_IGNORE_BATTERY_OPTIMIZATION_SETTINGS); 
  14.                 intent.setAction(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS); 
  15.                 intent.setData(Uri.parse("package:" + packageName)); 
  16.                 activity.startActivityForResult(intent, REQUEST_IGNORE_BATTERY_CODE); 
  17.             } 
  18.         } catch (Exception e) { 
  19.             e.printStackTrace(); 
  20.         } 
  21.     } 

在界面重写onActivityResult方法来捕获用户的选择。如,代码如下:

 
 
  1. @Override 
  2. protected void onActivityResult(int requestCode, int resultCode, Intent data) { 
  3.     if (resultCode == RESULT_OK) { 
  4.         if (requestCode == BatteryUtils.REQUEST_IGNORE_BATTERY_CODE){ 
  5.             //TODO something 
  6.         } 
  7.     }else if (resultCode == RESULT_CANCELED){ 
  8.         if (requestCode == BatteryUtils.REQUEST_IGNORE_BATTERY_CODE){ 
  9.             ToastUtils.show(getActivity(), "请开启忽略电池优化~"); 
  10.         } 
  11.     } 
  12.  

补充

当应用程序被Kill掉,但是闹钟的服务没有被Kill掉的,这时候又设置了闹钟。这就意味着设置的闹钟没有放到闹钟服务那里。所以这种情况,设置的闹钟会失效。为了解决这种情况,利用AIDL(闹钟服务在另一个进程的需要进程间通信)调用闹钟服务的重新设置闹钟方法重设闹钟。

在应用程序的onCreat()方法启动闹钟服务,然后再绑定闹钟服务。

 
 
  1. private void initAlarmService() { 
  2.     startService(new Intent(this, DaemonService.class));//启动闹钟服务 
  3.     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { 
  4.         //JobScheduler 
  5.         ... 
  6.     } 
  7.  
  8.     //绑定闹钟服务 
  9.     Intent intent = new Intent(this, DaemonService.class); 
  10.     intent.setAction("android.intent.action.DaemonService"); 
  11.     bindService(intent, mConnection, Context.BIND_AUTO_CREATE); 

在onDestroy()方法,调用闹钟服务的重设闹钟方法。代码如下:

 
 
  1. @Override 
  2. protected void onDestroy() { 
  3.     super.onDestroy(); 
  4.     try {//判断是否有闹钟,没有则关闭闹钟服务 
  5.         String alarm = localPreferencesHelper.getString(LocalPreferencesHelper.ALARM_CLOCK); 
  6.         if (daemonService != -1 && mIRemoteService != null) { 
  7. //                android.os.Process.killProcess(daemonService); 
  8.             mIRemoteService.resetAlarm(); 
  9.         } 
  10.          
  11.         if (!alarm.equals("[]")) { 
  12.             if (daemonService != -1) { 
  13.                 startService(new Intent(this, DaemonService.class)); 
  14.             } 
  15.         } else { 
  16.             if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { 
  17.                 mJobScheduler.cancel(JOB_ID); 
  18.             } 
  19.  
  20.         } 
  21.         unbindService(mConnection); //解除绑定服务。 
  22.     } catch (Exception e) { 
  23.  
  24.     } 
  25.  

这里说明一下,当服务启动并且被绑定的情况下,unbindService是不会停止服务的。具体可以查看这篇文章。here

最后

以上并不代表所有的Android手机的闹钟都可以用,这只是尽最大的可能保证大部分的手机。




作者:zhonghanwen
来源:51CTO
目录
相关文章
|
2月前
|
监控 安全 Android开发
【新手必读】Airtest测试Android手机常见的设置问题
【新手必读】Airtest测试Android手机常见的设置问题
|
6月前
|
数据库连接 数据库 Android开发
Android -- Room简化数据库设置图书案例
Android -- Room简化数据库设置图书案例
22 0
|
3天前
|
存储 Java Android开发
Android系统 设置第三方应用为默认Launcher实现和原理分析
Android系统 设置第三方应用为默认Launcher实现和原理分析
16 0
|
2天前
|
安全 编译器 API
Android HAL深入探索(5): 调试HAL报错与解决方案
Android HAL深入探索(5): 调试HAL报错与解决方案
6 1
|
1月前
|
JSON Android开发 数据格式
android 使用GSON 序列化对象出现字段被优化问题解决方案
android 使用GSON 序列化对象出现字段被优化问题解决方案
|
3月前
|
网络安全 Android开发
2023安卓逆向 -- 抓包环境设置(Charles+Postern)
2023安卓逆向 -- 抓包环境设置(Charles+Postern)
33 0
|
8月前
|
IDE Java 开发工具
Android Gradle plugin requires Java 11 to run. You are currently using Java 1.8的解决方案
Android Gradle plugin requires Java 11 to run. You are currently using Java 1.8的解决方案
|
8月前
|
XML Android开发 数据格式
Android中利用shape属性自定义设置Button按钮
Android中利用shape属性自定义设置Button按钮
137 0
|
4月前
|
XML 数据库 数据安全/隐私保护
Android App规范处理中版本设置、发布模式、给数据集SQLite加密的讲解及使用(附源码 超详细必看)
Android App规范处理中版本设置、发布模式、给数据集SQLite加密的讲解及使用(附源码 超详细必看)
42 0