Android学习之路(28) 进程保活组件的封装

前言

远古时代,出现过很多黑科技,比如MarsDaemon,使用双进程守护的方式进行保活,在当时可谓风光无限,可惜在8.0时代到来就被废弃了。

又比如后面出现的1像素Activity的保活方式,说他流氓一点不过分,如果每个人都使用这些操作,因为功耗的大大增加,会直接影响到手机使用寿命。 因此各大手机厂商为了解决这一现象的发生,在系统层面对这些流氓行为做出了限制: 后台进程即使你是要黑科技让进程优先级很高,也可能被杀死,所以我宁愿称是应用的求生而不是保活

这种方式初衷是好的,降低了设备功耗,降低了内存,防止手机发烫等,但是对于一些真正需要做保活操作的应用来说,可谓苦不堪言。

于是乎新型的求生措施又出现了。

如何优雅的进行求生

Android6.0以后系统推出了一个电池优化方案,对一些高耗电的进程会进行策略杀死。

那可能有人要问了,微信和qq这些高耗电应用是怎么做到保存的呢?

看下图

微信qq白名单.awebp

可以看到系统将微信和qq都放到了他们的白名单里面了,这个是怎么做到的呢?

其实, 这个是微信和厂商做了协商,将他们自己的应用设置到了他们的电量优化白名单中。 下次产品再问我为啥他们可以做到的时候,我就把这张图甩给他们。。

那我们可不可以也让厂商给我们加名单呢?呃呃呃。。

小伙子很有想法.webp

好在手机厂商没有把路堵死,给我们留了一条后路

  • 1.首先使用下面的代码检测我们进程是否在白名单中:
java
复制代码
@RequiresApi(api = Build.VERSION_CODES.M)
private boolean isIgnoringBatteryOptimizations() {boolean isIgnoring = false;PowerManager powerManager = (PowerManager) getSystemService(Context.POWER_SERVICE);if (powerManager != null) {isIgnoring = powerManager.isIgnoringBatteryOptimizations(getPackageName());}return isIgnoring;
}
  • 2.如果没有:调用下面的代码申请加入白名单
java
复制代码
@RequiresApi(api = Build.VERSION_CODES.M)
public void requestIgnoreBatteryOptimizations() {try {Intent intent = new Intent(Settings.ACTION_REQUEST_IGNORE_BATTERY_OPTIMIZATIONS);intent.setData(Uri.parse("package:" + getPackageName()));startActivity(intent);} catch (Exception e) {e.printStackTrace();}
}

申请时会弹出一个让用户选择的Dialog:

电池优化选项.awebp

窗口中会提示该操作可能是影响电池的使用,如果需要监听用户的按键,可以使用startActivityResultonActivityResult中监听

好了,白名单是加好了,那是不是就是万事大吉了呢?

手机厂商:哪有那么容易,就算你加入了电量优化白名单,你要是不按规矩来,在后台运行的进程还是会被我们杀掉?还有啥招式快快使出来吧

放大招了.gif eee。。

我们知道进程被杀死,是因为系统的后台管理系统把我们重启的路堵住了,为啥堵我啊?按我说可能系统看你这个进程不顺眼吧,哈哈。。

言归正传:

其实是你不在后台管理的自启动白名单中,自启动白名单就像一张通行证,你的应用需要在系统后台自启动,就要在白名单上,否则哪里来回哪里去吧

那白名单这么好,怎么才能加入TM呢?

要知道市面上手机厂家很多,每个厂家的系统都不一样,一个系统还有很多甚至几十个版本,这让我们怎么加入啊?

而大部分自启动操作可以在厂商的手机管家的设置里面设置:

最理想的做法:我们根据不同手机,甚至是不同的系统版本,给用户呈现一个图文操作步骤,并且提供一个按钮,直接跳转到指定页面进行设置

  • 我们先定义下面两个方法:
java
复制代码
/*** 跳转到指定应用的首页*/
private static void showActivity(Context context,@NonNull String packageName) {Intent intent = context.getPackageManager().getLaunchIntentForPackage(packageName);context.startActivity(intent);
}/*** 跳转到指定应用的指定页面*/
private void showActivity(Context context,@NonNull String packageName, @NonNull String activityDir) {Intent intent = new Intent();intent.setComponent(new ComponentName(packageName, activityDir));intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);context.startActivity(intent);
}
  • 手机厂商判断:
java
复制代码
华为:
public boolean isHuawei() {if (Build.BRAND == null) {return false;} else {return Build.BRAND.toLowerCase().equals("huawei") || Build.BRAND.toLowerCase().equals("honor");}
}
小米
public static boolean isXiaomi() {return Build.BRAND != null && Build.BRAND.toLowerCase().equals("xiaomi");
}
OPPO
public static boolean isOPPO() {return Build.BRAND != null && Build.BRAND.toLowerCase().equals("oppo");
}VIVO
public static boolean isVIVO() {return Build.BRAND != null && Build.BRAND.toLowerCase().equals("vivo");
}
魅族
public static boolean isMeizu() {return Build.BRAND != null && Build.BRAND.toLowerCase().equals("meizu");
}
三星
public static boolean isSamsung() {return Build.BRAND != null && Build.BRAND.toLowerCase().equals("samsung");
}
  • 手机管家或者自启动界面启动方式:
java
复制代码
华为:
private void goHuaweiSetting() {try {showActivity("com.huawei.systemmanager","com.huawei.systemmanager.startupmgr.ui.StartupNormalAppListActivity");} catch (Exception e) {showActivity("com.huawei.systemmanager","com.huawei.systemmanager.optimize.bootstart.BootStartActivity");}
}
小米:private void goXiaomiSetting() {showActivity("com.miui.securitycenter","com.miui.permcenter.autostart.AutoStartManagementActivity");
}OPPO:
private void goOPPOSetting() {try {showActivity("com.coloros.phonemanager");} catch (Exception e1) {try {showActivity("com.oppo.safe");} catch (Exception e2) {try {showActivity("com.coloros.oppoguardelf");} catch (Exception e3) {showActivity("com.coloros.safecenter");}}}
}VIVO
private void goVIVOSetting() {showActivity("com.iqoo.secure");
}魅族:
private void goMeizuSetting() {showActivity("com.meizu.safe");
}
三星:
private void goSamsungSetting() {try {showActivity("com.samsung.android.sm_cn");} catch (Exception e) {showActivity("com.samsung.android.sm");}
}

总结下上面我们所讲:

  • 1.为了不被电量优化,我们需要将应用添加进电量优化白名单中
  • 2.为了可以在被杀死后,自己可以启动自己,需要将应用自启动开关开启,可以使 用图文引导的方式:

参考下面这张图:

自启动.awebp

保活增强:

我们都知道保活操作一般是使用一个前台服务来挂起我们的应用: 还有的保活操作是使用一个JobService来对让系统在某个条件符合下回调一个请求操作。

基于以上分析:

  • 笔者这边封装了一个保活组件-lib_pull_alive: 结合了:前台服务+JobService+电量优化白名单+引导用户应用自启动的方式实现了一个求生方案,代码如下:
java
复制代码
keepAliveService.java
package com.anna.lib_keepalive.service;import android.app.job.JobInfo;
import android.app.job.JobParameters;
import android.app.job.JobScheduler;
import android.app.job.JobService;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.os.Build;
import android.os.Handler;
import android.os.Message;
import android.util.Log;import androidx.annotation.RequiresApi;import com.anna.lib_keepalive.forground.ForgroundNF;
import com.anna.lib_keepalive.utils.Utils;/*** 创建一个JobService用于提高应用优先级*/
@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
public class KeepAliveService extends JobService {private static final String TAG = KeepAliveService.class.getSimpleName();private JobScheduler mJobScheduler;private static final int JOB_ID = 1;private ComponentName JOB_PG;private int NOTIFICATION_ID = 10;private ForgroundNF mForgroundNF;private Handler mJobHandler = new Handler(new Handler.Callback() {@Overridepublic boolean handleMessage(Message msg) {Log.d(TAG, "pull alive.");jobFinished((JobParameters) msg.obj, true);return true;}});@Overridepublic void onCreate() {super.onCreate();mJobScheduler = (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);JOB_PG = new ComponentName(getPackageName(),KeepAliveService.class.getName());mForgroundNF = new ForgroundNF(this);Utils.requestIgnoreBatteryOptimizations(this);}@RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)public static void start(Context context){Intent intent = new Intent(context,KeepAliveService.class);context.startService(intent);}@Overridepublic boolean onStartJob(JobParameters params) {Log.d(TAG,"onStartJob");mJobHandler.sendMessage(Message.obtain(mJobHandler, 1, params));return true;}/**系统回调使用,说明触发了job条件* @param params* @return*/@Overridepublic boolean onStopJob(JobParameters params) {Log.d(TAG,"onStopJob");mJobHandler.sendEmptyMessage(1);return false;}@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {JobInfo job = initJob();mJobScheduler.schedule(job);startNotificationForGround();return START_STICKY;}/*** 大于18可以使用一个取消Notification的服务*/private void startNotificationForGround(){if(Build.VERSION.SDK_INT<18){mForgroundNF.startForegroundNotification();}else{mForgroundNF.startForegroundNotification();Intent it = new Intent(this, CancelNotifyervice.class);startService(it);}}/**初始化Job任务* @return*/private JobInfo initJob() {JobInfo.Builder builder = new JobInfo.Builder(JOB_ID, JOB_PG);if(Build.VERSION.SDK_INT>=Build.VERSION_CODES.N){builder.setMinimumLatency(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS); //执行的最小延迟时间builder.setOverrideDeadline(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS);  //执行的最长延时时间builder.setBackoffCriteria(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS,JobInfo.BACKOFF_POLICY_LINEAR);//线性重试方案}else {builder.setPeriodic(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS);}builder.setPersisted(false);builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_NONE);builder.setRequiresCharging(false);return builder.build();}@Overridepublic void onDestroy() {super.onDestroy();mForgroundNF.stopForegroundNotification();}}
java
复制代码
ForgroundNF.java
package com.anna.lib_keepalive.forground;import android.app.Notification;
import android.app.NotificationChannel;
import android.app.NotificationManager;
import android.app.Service;
import android.content.Context;
import android.os.Build;import androidx.core.app.NotificationCompat;import com.anna.lib_keepalive.R;public class ForgroundNF {private static final int START_ID = 101;private static final String CHANNEL_ID = "app_foreground_service";private static final String CHANNEL_NAME = "前台保活服务";private Service service;private NotificationManager notificationManager;private NotificationCompat.Builder mNotificationCompatBuilder;public ForgroundNF(Service service){this.service = service;initNotificationManager();initCompatBuilder();}/*** 初始化NotificationCompat.Builder*/private void initCompatBuilder() {mNotificationCompatBuilder = new NotificationCompat.Builder(service,CHANNEL_ID);//标题mNotificationCompatBuilder.setContentTitle("test keep alive");//通知内容mNotificationCompatBuilder.setContentText("test alive");mNotificationCompatBuilder.setSmallIcon(R.mipmap.ic_launcher_round);}/*** 初始化notificationManager并创建NotificationChannel*/private void initNotificationManager(){notificationManager = (NotificationManager) service.getSystemService(Context.NOTIFICATION_SERVICE);//针对8.0+系统if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {NotificationChannel channel  = new NotificationChannel(CHANNEL_ID,CHANNEL_NAME,NotificationManager.IMPORTANCE_LOW);channel.setLockscreenVisibility(Notification.VISIBILITY_PUBLIC);channel.setShowBadge(false);notificationManager.createNotificationChannel(channel);}}public void startForegroundNotification(){service.startForeground(START_ID,mNotificationCompatBuilder.build());}public void stopForegroundNotification(){notificationManager.cancelAll();service.stopForeground(true);}}

完整代码可以查看github上的Demo: github.com/ByteYuhb/an…

总结:

本文是组件化开发的第四篇,也是第三个功能组件的封装,都已上传到Github,后期会陆续推荐其他组件的封装。

组件化开发.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/251633.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

自动运维ansible实训(网络管理与维护综合实训)

来自即将退役学长的分享&#xff0c;祝学弟学妹以后发大财&#xff01; 一 实训目的及意义 1.1 实训目的 1、熟悉自动化运维工具&#xff1a;实训旨在让学员熟悉 Ansible 这一自动化运维工具。通过实际操作&#xff0c;学员可以了解 Ansible 的基本概念、工作原理和使用方法…

【数据结构】排序---C语言版

七大排序算法 一、对于排序的分类&#xff1a;二、插入排序1、直接插入排序&#xff08;1&#xff09;基本思想&#xff1a;&#xff08;2&#xff09;直接插入排序&#xff1a;&#xff08;3&#xff09;代码实现&#xff1a;&#xff08;4&#xff09;总结&#xff1a; 2、希…

BAPI创建会计凭证和冲销凭证

目录 BAPI创建会计凭证和冲销凭证组件 利润中心冲销不可能原因由于一个变量为空导致不可以冲销 代码附上创建会计凭证代码冲销会计凭证代码 BAPI创建会计凭证和冲销凭证 在使用冲销会计凭证的BAPI&#xff0c;即BAPI_ACC_DOCUMENT_REV_POST时&#xff0c;遇到下面的问题&#…

Vue(二十):ElementUI 扩展实现表格组件的拖拽行

效果 源码 注意&#xff1a; 表格组件必须添加 row-key 属性&#xff0c;用来优化表格的渲染 <template><el-row :gutter"10"><el-col :span"12"><el-card class"card"><el-scrollbar><span>注意: 表格组件…

【数据库】关系型和非关系型数据库的区别?

&#x1f34e;个人博客&#xff1a;个人主页 &#x1f3c6;个人专栏&#xff1a;JAVA ⛳️ 功不唐捐&#xff0c;玉汝于成 目录 正文 关系型数据库的优点 非关系型数据库&#xff08;NOSQL&#xff09;的优点 我的其他博客 正文 关系型数据库的优点 容易理解&#xff0c…

Python 数据可视化:配色方案

1、引言 在这篇文章中&#xff0c;我们将研究Python的一些配色方案&#xff0c;主要是Seaborn库。这将采用 Python Notebook 格式&#xff0c;其中包括绘图的代码。 2、实验数据 首先导入必要的库&#xff1a; import pandas as pd import seaborn as sns import matplotlib…

基于springboot篮球竞赛预约平台源码和论文

随着信息化时代的到来&#xff0c;管理系统都趋向于智能化、系统化&#xff0c;篮球竞赛预约平台也不例外&#xff0c;但目前国内仍都使用人工管理&#xff0c;市场规模越来越大&#xff0c;同时信息量也越来越庞大&#xff0c;人工管理显然已无法应对时代的变化&#xff0c;而…

虚幻UE 特效-Niagara特效实战-魔法阵

回顾Niagara特效基础知识&#xff1a;虚幻UE 特效-Niagara特效初识 其他四篇实战&#xff1a;UE 特效-Niagara特效实战-烟雾、喷泉、 虚幻UE 特效-Niagara特效实战-火焰、烛火、 虚幻UE 特效-Niagara特效实战-雨天、 虚幻UE 特效-Niagara特效实战-眩晕。 本篇笔记记录了使用空模…

Matplotlib热力图的创意绘制指南【第54篇—python:Matplotlib热力图】

文章目录 Matplotlib热力图的创意绘制指南1. 简介2. 基本热力图3. 自定义颜色映射4. 添加注释5. 不同形状的热力图6. 分块热力图7. 多子图热力图8. 3D热力图9. 高级颜色映射与颜色栏设置10. 热力图的动态展示11. 热力图的交互性12. 标准化数据范围13. 导出热力图 总结&#xff…

动态颗粒背景,适合VUE、HTML前端显示

动态颗粒背景&#xff0c;适合做背景使用&#xff0c;VUE、HTML前端显示直接看效果 废话不多说直接上代码&#xff1b; 一、html 代码部分 <template><div id"login"><div class"container"><div class"login-form"&g…

C++实战Opencv第二天——色彩空间转换函数和opencv中图像对象创建与赋值(从零开始,保姆教学)

OpenCV是一个强大的计算机视觉库&#xff0c;使用C作为主要编程语言&#xff0c;对于图像处理和计算机视觉领域具有重要意义。其提供了丰富的功能和算法&#xff0c;使得开发者能够快速实现各种图像处理和计算机视觉应用。OpenCV C为图像处理和计算机视觉领域的开发者提供了一个…

[Vue3]父子组件相互传值数据同步

简介 vue3中使用setup语法糖&#xff0c;父子组件之间相互传递数据及数据同步问题 文章目录 简介父传子props传递值 使用v-bind绑定props需要计算toRefcomputed emit传递方法 使用v-on绑定 子传父expose v-model总结 父传子 props传递值 使用v-bind绑定 父组件通过props给子…

【前沿技术杂谈:开源软件】引领技术创新与商业模式的革命

【前沿技术杂谈&#xff1a;开源软件】引领技术创新与商业模式的革命 开源软件如何推动技术创新开源软件的开放性和协作精神促进知识共享和技术迭代推动关键技术的发展开源软件与新技术的融合 开源软件的商业模式开源软件的商业模式将开源软件与商业软件相结合 开源软件的安全风…

三维可视化助力船舶制造:大数据处理、实时协作更高效!

随着科技的不断发展&#xff0c;船舶制造行业也在不断寻求创新和提高效率的途径。其中&#xff0c;HOOPS技术作为一种先进的三维可视化和工程协作技术&#xff0c;正逐渐成为船舶制造领域的关键工具。 本文将深入探讨HOOPS技术在船舶制造行业的应用&#xff0c;探讨其带来的优…

张维迎《博弈与社会》威胁与承诺(4)宪政与民主

有限政府 动态博弈理论对我们理解民主与法治具有重要的意义。 自人类进入文明时代以来&#xff0c;政府就是社会博弈重要的参与人。任何社会要有效运行&#xff0c;都需要赋予政府一些自由裁量权。但如果政府的自由裁量权太大&#xff0c;政府官员为所欲为&#xff0c;不仅老百…

Python详细教程

一、Python简历 Python 是一个高层次的结合了解释性、编译性、互动性和面向对象的脚本语言。 Python 的设计具有很强的可读性&#xff0c;相比其他语言经常使用英文关键字&#xff0c;其他语言的一些标点符号&#xff0c;它具有比其他语言更有特色语法结构。 Python 是一种解…

flask+pyinstaller实现mock接口,并打包到exe运行使用postman验证

flask代码 from flask import Flask, request, jsonifyapp Flask(__name__)app.route("/login", methods[POST]) def login():username request.json.get("username").strip() # 用户名password request.json.get("password").strip() # 密…

【从0上手Cornerstone3D】如何使用CornerstoneTools中的工具之工具介绍

简单介绍一下在Cornerstone中什么是工具&#xff0c;工具是一个未实例化的类&#xff0c;它至少实现了BaseTool接口。 如果我们想要在我们的代码中使用一个工具&#xff0c;则必须实现以下两个步骤&#xff1a; 使用Cornerstone的顶层addTool函数添加未实例化的工具 将工具添…

小白水平理解面试经典题目LeetCode 21. Merge Two Sorted Lists【Linked List类】

21. 将两个有序列表融合 Linked List 数据结构也在面试中经常出现&#xff0c;作为很好处理客户信息存储的结构很方便&#xff0c;也是重点必会项目之一&#xff0c;看看我们如何教懂白月光&#xff0c;成功邀约看电影吧。 小白渣翻译 你将获得两个排序链表 list1 和 list2 …

【蓝桥杯选拔赛真题64】python数字塔 第十五届青少年组蓝桥杯python 选拔赛比赛真题解析

python数字塔 第十五届蓝桥杯青少年组python比赛选拔赛真题 一、题目要求 (注:input()输入函数的括号中不允许添加任何信息) 提示信息: 数字塔是由 N 行数堆积而成,最顶层只有一个数,次顶层两个数,以此类推。相邻层之间的数用线连接,下一层的每个数与它上一层左上…