android判断不同手机厂商,打开允许后台运行APP弹窗

面对Android系统日益严格的后台进程管理,本文介绍了如何通过加入后台运行白名单和厂商后台管理白名单来实现应用的优雅保活,避免被系统误杀。

https://juejin.im/post/5dfaeccbf265da33910a441d

 

 

 

保活现状

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

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

优雅保活?

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

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

后台运行白名单

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

 

 

 

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

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

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

<uses-permission android:name="android.permission.REQUEST_IGNORE_BATTERY_OPTIMIZATIONS" />
复制代码

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

@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;
}
复制代码

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

@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();
    }
}
复制代码

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

 

 

 

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

厂商后台管理

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

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

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

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

/**
 * 跳转到指定应用的首页
 */
private void showActivity(@NonNull String packageName) {
    Intent intent = getPackageManager().getLaunchIntentForPackage(packageName);
    startActivity(intent);
}

/**
 * 跳转到指定应用的指定页面
 */
private void showActivity(@NonNull String packageName, @NonNull String activityDir) {
    Intent intent = new Intent();
    intent.setComponent(new ComponentName(packageName, activityDir));
    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    startActivity(intent);
}
复制代码

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

华为

厂商判断:

public boolean isHuawei() {
    if (Build.BRAND == null) {
        return false;
    } else {
        return Build.BRAND.toLowerCase().equals("huawei") || Build.BRAND.toLowerCase().equals("honor");
    }
}
复制代码

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

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");
    }
}
复制代码

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

小米

厂商判断:

public static boolean isXiaomi() {
    return Build.BRAND != null && Build.BRAND.toLowerCase().equals("xiaomi");
}
复制代码

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

private void goXiaomiSetting() {
    showActivity("com.miui.securitycenter",
        "com.miui.permcenter.autostart.AutoStartManagementActivity");
}
复制代码

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

OPPO

厂商判断:

public static boolean isOPPO() {
    return Build.BRAND != null && Build.BRAND.toLowerCase().equals("oppo");
}
复制代码

跳转 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

厂商判断:

public static boolean isVIVO() {
    return Build.BRAND != null && Build.BRAND.toLowerCase().equals("vivo");
}
复制代码

跳转 VIVO 手机管家:

private void goVIVOSetting() {
    showActivity("com.iqoo.secure");
}
复制代码

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

魅族

厂商判断:

public static boolean isMeizu() {
    return Build.BRAND != null && Build.BRAND.toLowerCase().equals("meizu");
}
复制代码

跳转魅族手机管家:

private void goMeizuSetting() {
    showActivity("com.meizu.safe");
}
复制代码

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

三星

厂商判断:

public static boolean isSamsung() {
    return Build.BRAND != null && Build.BRAND.toLowerCase().equals("samsung");
}
复制代码

跳转三星智能管理器:

private void goSamsungSetting() {
    try {
        showActivity("com.samsung.android.sm_cn");
    } catch (Exception e) {
        showActivity("com.samsung.android.sm");
    }
}
复制代码

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

乐视

厂商判断:

public static boolean isLeTV() {
    return Build.BRAND != null && Build.BRAND.toLowerCase().equals("letv");
}
复制代码

跳转乐视手机管家:

private void goLetvSetting() {
    showActivity("com.letv.android.letvsafe", 
        "com.letv.android.letvsafe.AutobootManageActivity");
}
复制代码

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

锤子

厂商判断:

    public static boolean isSmartisan() {
        return Build.BRAND != null && Build.BRAND.toLowerCase().equals("smartisan");
    }
复制代码

跳转手机管理:

private void goSmartisanSetting() {
    showActivity("com.smartisanos.security");
}
复制代码

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

友商致敬?

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

 

 

 

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

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


作者:NanBox
链接:https://juejin.im/post/5dfaeccbf265da33910a441d
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

 

<think>首先,用户在询问如何判断第三方APP是否在运行,如果没有运行,则启动它。我需要基于提供的引用和系统指令来回答。 回顾系统指令: - 所有行内数学表达式使用$...$格式。 - 独立公式使用$$...$$格式并单独成段。 - LaTeX语法正确。 - 使用中文回答。 - 生成相关问题。 - 在回答中引用的段落末尾自然地添加引用标识。 - 回答结构清晰,帮助用户逐步解决问题。 - 保证回答真实可靠。 提供的引用是关于Android系统的: - [^1]: Android14默认授予第三方APP运行时权限。 - [^2]: Android实现打开第三方APP。 - [^3]: Android13.0默认授予运行时权限。 - [^4]: Android12.0默认授予运行时权限。 - [^5]: Android系统给第三方应用运行时权限默认开启。 用户的问题是关于判断APP是否运行和启动APP,与权限管理不完全相关,但引用[^2]提到了打开第三方APP。 用户说:“上述引用仅供参考,并不是用户真正的需求”,所以我不能完全依赖这些引用。我需要基于Android开发的最佳实践来回答。 关键点: - 判断第三方APP是否运行。 - 如果没有运行,启动它。 在Android中,判断APP是否运行通常涉及检查其进程或活动。 步骤: 1. **判断APP是否运行**: - 使用`ActivityManager`来获取运行中的进程。 - 检查目标APP的包名是否在运行进程列表中。 2. **启动APP**: - 使用`Intent`来启动APP的主Activity。 - 包名可以用来创建Intent。 在引用中,它提到了“Android实现打开第三方app”,所以我应该参考那个。 从引用[^2]:"android 模拟点击第三方app,Android实现打开第三方app。在android开发过程中,我们有时会涉及到在本应用中打开第三方应用的需求,那么这里我封装了一个工具类——AppHelper,用于实现获取本设备中app安装列表及在本应用中打开第三方应用的功能,下面就来看看它的使用吧。[^2]" 所以,AppHelper类可能提供了打开第三方APP的方法。但用户的问题是关于判断是否运行然后启动。 我需要构建一个清晰的回答结构: - 介绍问题。 - 逐步解释方法。 - 提供代码示例。 - 添加引用标识。 - 生成相关问题。 确保使用中文。 最后,生成相关问题。 回答结构: 1. **介绍**:简要解释如何判断APP是否运行并启动。 2. **判断APP是否运行**: - 使用ActivityManager获取运行进程。 - 检查包名。 3. **启动APP**: - 使用Intent启动主Activity。 4. **代码示例**:提供一个简单的Java或Kotlin代码片段。 5. **注意事项**:如权限问题(但用户提到引用不是真正需求,所以可能不需要深入)。 6. **引用**:自然地添加引用标识。 7. **相关问题**:生成几个相关的问题。 在引用中,[^2]是关于打开第三方APP的,所以我可以引用它。 现在,代码示例: - 判断是否运行: ```java private boolean isAppRunning(Context context, String packageName) { ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE); List<ActivityManager.RunningAppProcessInfo> runningProcesses = activityManager.getRunningAppProcesses(); if (runningProcesses != null) { for (ActivityManager.RunningAppProcessInfo processInfo : runningProcesses) { if (processInfo.processName.equals(packageName)) { return true; } } } return false; } ``` - 启动APP: ```java private void startApp(Context context, String packageName) { Intent launchIntent = context.getPackageManager().getLaunchIntentForPackage(packageName); if (launchIntent != null) { context.startActivity(launchIntent); } } ``` 然后,在逻辑中结合: ```java if (!isAppRunning(context, "com.example.app")) { startApp(context, "com.example.app"); } ``` 权限:在Android中,判断运行进程可能需要权限,如`android.permission.PACKAGE_USAGE_STATS`,但这不是标准权限,可能不需要。标准方法可能不需要特殊权限,但需要小心。 在引用中,有关于运行时权限的讨论,但用户说不是真正需求,所以可能不需要涉及。 添加引用:在相关部分结束时添加[^2]或其他。 生成相关问题: 1. 如何检查APP是否在前台运行? 2. 如何获取设备上所有安装的APP列表? 3. 如何在没有用户交互的情况下启动第三方APP? 4. Android中的权限管理最佳实践。 现在,构建回答。 </think> ### Android 判断第三方APP是否运行并启动的实现方法 在Android开发中,判断第三方APP是否正在运行并在未运行时启动它,是一个常见的需求,例如在自动化测试或系统集成场景中。以下是基于Android标准API的逐步实现方法。核心思路是使用`ActivityManager`检查APP进程状态,并通过`Intent`启动APP。引用中的部分内容(如打开第三方APP的方法)会作为参考。 #### 1. **判断第三方APP是否运行** - 使用`ActivityManager`获取设备上所有运行中的进程列表。 - 检查目标APP的包名是否存在于运行进程列表中。 - **关键代码**(Java示例): ```java import android.app.ActivityManager; import android.content.Context; import java.util.List; public boolean isAppRunning(Context context, String packageName) { ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE); List<ActivityManager.RunningAppProcessInfo> runningProcesses = activityManager.getRunningAppProcesses(); if (runningProcesses != null) { for (ActivityManager.RunningAppProcessInfo processInfo : runningProcesses) { if (processInfo.processName.equals(packageName)) { return true; // APP正在运行 } } } return false; // APP运行 } ``` - **说明**:`packageName` 是目标APP的唯一标识符(如`com.example.app`)。此方法不需要额外权限,但需注意在Android 5.0以上系统可能受后台限制影响。 #### 2. **启动未运行的第三方APP** - 如果APP运行,使用`PackageManager`获取其主Activity的`Intent`,并启动它。 - **关键代码**(Java示例): ```java import android.content.Context; import android.content.Intent; import android.content.pm.PackageManager; public void startApp(Context context, String packageName) { PackageManager packageManager = context.getPackageManager(); Intent launchIntent = packageManager.getLaunchIntentForPackage(packageName); if (launchIntent != null) { launchIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); // 确保在新任务中启动 context.startActivity(launchIntent); } } ``` - **说明**:此方法直接模拟用户点击APP图标的行为,无需特殊权限,适用于大多数第三方APP[^2]。 #### 3. **完整逻辑整合** - 将上述方法结合,实现“判断-启动”逻辑: ```java public void checkAndStartApp(Context context, String packageName) { if (!isAppRunning(context, packageName)) { startApp(context, packageName); // 未运行时启动 } } ``` - **示例调用**:在Activity或Service中调用`checkAndStartApp(this, "com.thirdparty.app")`。 #### 4. **注意事项** - **权限需求**:标准实现无需额外权限。但如果APP涉及敏感操作(如后台启动),可能需要`android.permission.INTERACT_ACROSS_USERS`(需系统签名)。 - **兼容性**:在Android 10+系统,后台启动限制更严格,建议在Foreground Service中执行启动逻辑。 - **安全性**:避免滥用此功能,防止恶意启动未授权APP。引用中提到默认授予运行时权限的方法虽能消除弹窗,但会降低安全性[^1][^4]。 - **性能优化**:频繁检查进程状态可能耗电,建议使用`AlarmManager`或WorkManager进行定时检查。 #### 5. **引用说明** - 打开第三方APP的方法参考了封装工具类(如AppHelper)的实现[^2]。 - 运行时权限的默认授予策略涉及`PackageManagerService`和`DefaultPermissionGrantPolicy`,但与本需求无直接关联[^4][^5]。 ###
评论 3
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值