欢迎访问移动开发之家(rcyd.net),关注移动开发教程。移动开发之家  移动开发问答|  每日更新
页面位置 : > > > 内容正文

实现后台优雅保活有困难?不存在的!,

来源: 开发者 投稿于  被查看 34667 次 评论:281

实现后台优雅保活有困难?不存在的!,


保活现状

我们知道,Android 系统会存在杀后台进程的情况,并且随着系统版本的更新,杀进程的力度还有越来越大的趋势。系统这种做法本身出发点是好的,因为可以节省内存,降低功耗,也避免了一些流氓行为。

但有一部分应用,应用本身的使用场景就需要在后台运行,用户也是愿意让它在后台运行的,比如跑步类应用。一方面流氓软件用各种流氓手段进行保活,另一方面系统加大杀后台的力度,导致我们一些真正需要在后台运行的应用被误杀,苦不堪言。

优雅保活?

为了做到保活,出现了不少「黑科技」,比如 1 个像素的 Activity,播放无声音频,双进程互相守护等。这些做法可以说是很流氓了,甚至破坏了 Android 的生态,好在随着 Android 系统版本的更新,这些非常规的保活手段很多都已失效了。

对于那些确实需要在后台运行的应用,我们如何做到优雅的保活呢?

后台运行白名单

从 Android 6.0 开始,系统为了省电增加了休眠模式,系统待机一段时间后,会杀死后台正在运行的进程。但系统会有一个后台运行白名单,白名单里的应用将不会受到影响,在原生系统下,通过「设置」 - 「电池」 - 「电池优化」 - 「未优化应用」,可以看到这个白名单,通常会看到下面这两位:

下次被产品说「 XXX 都可以保活,为什么我们不行!」的时候,你就知道怎么怼回去了。大厂通过和手机厂商的合作,将自己的应用默认加入到白名单中。如果你在一个能谈成这种合作的大厂,也就不用往下看了。

好在系统还没有抛弃我们,允许我们申请把应用加入白名单。

首先,在 AndroidManifest.xml 文件中配置一下权限:

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

可以通过以下方法,判断我们的应用是否在白名单中:

  1. @RequiresApi(api = Build.VERSION_CODES.M) 
  2. private boolean isIgnoringBatteryOptimizations() { 
  3.     boolean isIgnoring = false; 
  4.     PowerManager powerManager = (PowerManager) getSystemService(Context.POWER_SERVICE); 
  5.     if (powerManager != null) { 
  6.         isIgnoring = powerManager.isIgnoringBatteryOptimizations(getPackageName()); 
  7.     } 
  8.     return isIgnoring; 

如果不在白名单中,可以通过以下代码申请加入白名单:

  1. @RequiresApi(api = Build.VERSION_CODES.M) 
  2. public void requestIgnoreBatteryOptimizations() { 
  3.     try { 
  4.         Intent intent = new Intent(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS); 
  5.         intent.setData(Uri.parse("package:" + getPackageName())); 
  6.         startActivity(intent); 
  7.     } catch (Exception e) { 
  8.         e.printStackTrace(); 
  9.     } 

申请时,应用上会出现这样一个窗口:

可以看到,这个系统弹窗会有影响电池续航的提醒,所以如果想让用户点允许,必须要有相关的说明。如果要判断用户是否点击了允许,可以在申请的时候调用 startActivityForResult,在 onActivityResult 里再判断一次是否在白名单中。

厂商后台管理

Android 开发的一个难点在于,各大手机厂商对原生系统进行了不同的定制,导致我们需要进行不同的适配,后台管理就是一个很好的体现。几乎各个厂商都有自己的后台管理,就算应用加入了后台运行白名单,仍然可能会被厂商自己的后台管理干掉。

如果能把应用加入厂商系统的后台管理白名单,可以进一步降低进程被杀的概率。不同的厂商在不同的地方进行设置,一般是在各自的「手机管家」,但更难的是,就算同一个厂商的系统,不同的版本也可能是在不同地方设置。

最理想的做法是,我们根据不同手机,甚至是不同的系统版本,给用户呈现一个图文操作步骤,并且提供一个按钮,直接跳转到指定页面进行设置。但需要对每个厂商每个版本进行适配,工作量是比较大的。我使用真机测试了大部分主流 Android 厂商的手机后,整理出了部分手机的相关资料。

首先我们可以定义这样两个方法:

  1. /** 
  2.  * 跳转到指定应用的首页 
  3.  */ 
  4. private void showActivity(@NonNull String packageName) { 
  5.     Intent intent = getPackageManager().getLaunchIntentForPackage(packageName); 
  6.     startActivity(intent); 
  7.  
  8. /** 
  9.  * 跳转到指定应用的指定页面 
  10.  */ 
  11. private void showActivity(@NonNull String packageName, @NonNull String activityDir) { 
  12.     Intent intent = new Intent(); 
  13.     intent.setComponent(new ComponentName(packageName, activityDir)); 
  14.     intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); 
  15.     startActivity(intent); 

以下是部分手机的厂商判断,跳转方法及对应设置步骤,跳转方法不保证在所有版本上都能成功跳转,都需要加 try catch。

华为

厂商判断:

  1. public boolean isHuawei() { 
  2.     if (Build.BRAND == null) { 
  3.         return false; 
  4.     } else { 
  5.         return Build.BRAND.toLowerCase().equals("huawei") || Build.BRAND.toLowerCase().equals("honor"); 
  6.     } 

跳转华为手机管家的启动管理页:

  1. private void goHuaweiSetting() { 
  2.     try { 
  3.         showActivity("com.huawei.systemmanager", 
  4.             "com.huawei.systemmanager.startupmgr.ui.StartupNormalAppListActivity"); 
  5.     } catch (Exception e) { 
  6.         showActivity("com.huawei.systemmanager", 
  7.             "com.huawei.systemmanager.optimize.bootstart.BootStartActivity"); 
  8.     } 

操作步骤:应用启动管理 -> 关闭应用开关 -> 打开允许自启动

小米

厂商判断:

  1. public static boolean isXiaomi() { 
  2.     return Build.BRAND != null && Build.BRAND.toLowerCase().equals("xiaomi"); 

跳转小米安全中心的自启动管理页面:

  1. private void goXiaomiSetting() { 
  2.     showActivity("com.miui.securitycenter", 
  3.         "com.miui.permcenter.autostart.AutoStartManagementActivity"); 

操作步骤:授权管理 -> 自启动管理 -> 允许应用自启动

OPPO

厂商判断:

  1. public static boolean isOPPO() { 
  2.     return Build.BRAND != null && Build.BRAND.toLowerCase().equals("oppo"); 

跳转 OPPO 手机管家:

  1. private void goOPPOSetting() { 
  2.     try { 
  3.         showActivity("com.coloros.phonemanager"); 
  4.     } catch (Exception e1) { 
  5.         try { 
  6.             showActivity("com.oppo.safe"); 
  7.         } catch (Exception e2) { 
  8.             try { 
  9.                 showActivity("com.coloros.oppoguardelf"); 
  10.             } catch (Exception e3) { 
  11.                 showActivity("com.coloros.safecenter"); 
  12.             } 
  13.         } 
  14.     } 

操作步骤:权限隐私 -> 自启动管理 -> 允许应用自启动

VIVO

厂商判断:

  1. public static boolean isVIVO() { 
  2.     return Build.BRAND != null && Build.BRAND.toLowerCase().equals("vivo"); 

跳转 VIVO 手机管家:

  1. private void goVIVOSetting() { 
  2.     showActivity("com.iqoo.secure"); 

操作步骤:权限管理 -> 自启动 -> 允许应用自启动

魅族

厂商判断:

  1. public static boolean isMeizu() { 
  2.     return Build.BRAND != null && Build.BRAND.toLowerCase().equals("meizu"); 

跳转魅族手机管家:

  1. private void goMeizuSetting() { 
  2.     showActivity("com.meizu.safe"); 

操作步骤:权限管理 -> 后台管理 -> 点击应用 -> 允许后台运行

三星

厂商判断:

  1. public static boolean isSamsung() { 
  2.     return Build.BRAND != null && Build.BRAND.toLowerCase().equals("samsung"); 

跳转三星智能管理器:

  1. private void goSamsungSetting() { 
  2.     try { 
  3.         showActivity("com.samsung.android.sm_cn"); 
  4.     } catch (Exception e) { 
  5.         showActivity("com.samsung.android.sm"); 
  6.     } 

操作步骤:自动运行应用程序 -> 打开应用开关 -> 电池管理 -> 未监视的应用程序 -> 添加应用

乐视

厂商判断:

  1. public static boolean isLeTV() { 
  2.     return Build.BRAND != null && Build.BRAND.toLowerCase().equals("letv"); 

跳转乐视手机管家:

  1. private void goLetvSetting() { 
  2.     showActivity("com.letv.android.letvsafe", 
  3.         "com.letv.android.letvsafe.AutobootManageActivity"); 

操作步骤:自启动管理 -> 允许应用自启动

锤子

厂商判断:

  1. public static boolean isSmartisan() { 
  2.     return Build.BRAND != null && Build.BRAND.toLowerCase().equals("smartisan"); 

跳转手机管理:

  1. private void goSmartisanSetting() { 
  2.     showActivity("com.smartisanos.security"); 

操作步骤:权限管理 -> 自启动权限管理 -> 点击应用 -> 允许被系统启动

友商致敬?

在之前做的跑步应用中,我在设置里增加了一个权限设置页面,将上面提到的设置放在这里面。最近发现友商某咚也跟进了,图 1 是我们做的,图 2 是某咚做的:


某咚从设计、从我写的不够好的文案,甚至是我从十几台手机上一张一张截下来的图,进行了全方位的致敬。感谢某咚的认可,但最近在某个发布会上听到这么一句话:在致敬的同时,能不能说一句谢谢?

某咚的致敬,一方面说明了目前确实存在进程容易被杀,保活难度大的问题,另一方面也说明了这种引导用户进行白名单设置的手段是有效的。

用户评论