Android進程保活之提升進程優(yōu)先級
一、1 像素 Activity 提高進程優(yōu)先級
使用 Activity 可以提升進程的 oom_adj 值 ;
APP 進入后臺后 , 使用 BroadcastReceiver 廣播接收者 , 監(jiān)聽 Android 系統(tǒng)的鎖屏廣播事件 ;
- 屏幕鎖定 : 啟動只有 1 1 1 像素的透明 Activity 界面 ;
- 屏幕解鎖 : 退出上述 1 1 1 像素的透明 Activity 界面 ;
1、主界面 MainActivity
主界面 , 主要負責注冊廣播接收者 ;
package kim.hsl.keep_progress_alive;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 注冊廣播接收者
KeepProgressAliveManager.getmInstance().registerReceiver(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
// 取消注冊廣播接收者, 也可以不取消注冊
//KeepProgressAliveManager.getmInstance().registerReceiver(this);
}
}
2、1 像素 Activity
在鎖屏?xí)r , 彈出的 1 像素 Activity , 有可能有進程?;畹耐?, 也彈出個同樣類型的 Activity , 一般都是透明的 , 即使這樣 , 最次也是個可見進程 ;
package kim.hsl.keep_progress_alive;
import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.Gravity;
import android.view.Window;
import android.view.WindowManager;
import androidx.annotation.Nullable;
/**
* 只有 1 像素的 Activity
*/
public class OnePixelActivity extends Activity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.i("OnePixelActivity", "onCreate");
// 獲取本界面的窗口 Window 對象
Window window = getWindow();
// 屏幕左上角展示
window.setGravity(Gravity.LEFT | Gravity.TOP);
// 將 Activity 設(shè)置成 1 像素
WindowManager.LayoutParams layoutParams = window.getAttributes();
// 寬高都設(shè)置 1 像素
layoutParams.width = 1;
layoutParams.height = 1;
// 放置位置 (0, 0) 坐標開始放置
layoutParams.x = 0;
layoutParams.y = 0;
// 在將布局參數(shù)設(shè)置會 Window 對象中
window.setAttributes(layoutParams);
// 設(shè)置界面到 KeepProgressAliveManager 單例對象中 , 用于關(guān)閉界面
KeepProgressAliveManager.getmInstance().setmOnePixelActivity(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
Log.i("OnePixelActivity", "onDestroy");
}
}
3、廣播接收者
監(jiān)聽 Intent.ACTION_SCREEN_OFF 和 Intent.ACTION_SCREEN_ON , 兩個廣播 , 再鎖屏?xí)r啟動 1 像素 Activity , 在解除鎖屏?xí)r , 關(guān)閉 1 像素 Activity ;
package kim.hsl.keep_progress_alive;
import android.annotation.SuppressLint;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
public class KeepProgressAliveReceiver extends BroadcastReceiver {
@SuppressLint("LongLogTag")
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
Log.i("KeepProgressAliveReceiver", "action : " + action);
if (Intent.ACTION_SCREEN_OFF.equals(action)){
// 鎖屏?xí)r開啟 Activity
KeepProgressAliveManager.getmInstance().startActivity(context);
Log.i("KeepProgressAliveReceiver", "鎖屏, 開啟 1 像素 Activity");
}else if (Intent.ACTION_SCREEN_ON.equals(action)){
// 解除所屏蔽關(guān)閉 Activity
KeepProgressAliveManager.getmInstance().finishActivity();
Log.i("KeepProgressAliveReceiver", "解除鎖屏, 關(guān)閉 1 像素 Activity");
}
}
}
4、管理類
單例管理類 , 負責注冊廣播接收者 , 在廣播接收者中啟動 1 像素頁面 , 同時也負責關(guān)閉該 1 像素頁面 ;
該管理類負責 Activity 組件與 BroadcastReceiver 組件的耦合 ;
package kim.hsl.keep_progress_alive;
import android.annotation.SuppressLint;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.util.Log;
public class KeepProgressAliveManager {
private static KeepProgressAliveManager mInstance;
private KeepProgressAliveManager(){}
public static KeepProgressAliveManager getmInstance(){
if (mInstance == null){
mInstance = new KeepProgressAliveManager();
}
return mInstance;
}
/**
* 注冊的廣播接收者
*/
private KeepProgressAliveReceiver mKeepProgressAliveReceiver;
/**
* 打開的 1 像素 Activity 界面
*/
private OnePixelActivity mOnePixelActivity;
/**
* 注冊廣播接收者
* @param context
*/
@SuppressLint("LongLogTag")
public void registerReceiver(Context context){
Log.i("KeepProgressAliveManager", "注冊廣播接收者");
IntentFilter intentFilter = new IntentFilter();
// 監(jiān)聽屏幕解除鎖定廣播
intentFilter.addAction(Intent.ACTION_SCREEN_ON);
// 監(jiān)聽[屏幕鎖定廣播
intentFilter.addAction(Intent.ACTION_SCREEN_OFF);
// 創(chuàng)建廣播接收者
mKeepProgressAliveReceiver = new KeepProgressAliveReceiver();
// 注冊廣播接收者
context.registerReceiver(mKeepProgressAliveReceiver, intentFilter);
}
/**
* 解除廣播接收者注冊
*/
@SuppressLint("LongLogTag")
public void unregisterReceiver(Context context){
Log.i("KeepProgressAliveManager", "取消注冊廣播接收者");
if(mKeepProgressAliveReceiver != null){
context.unregisterReceiver(mKeepProgressAliveReceiver);
mKeepProgressAliveReceiver = null;
}
}
/**
* 開啟 Activity 界面
*/
public void startActivity(Context context){
// 開啟 OnePixelActivity 界面
Intent intent = new Intent(context, OnePixelActivity.class);
// 重新創(chuàng)建一個任務(wù)棧 ( 前提是親和性不同 )
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
}
/**
* 設(shè)置 1 像素 Activity 界面, 用于關(guān)閉
* @param mOnePixelActivity
*/
public void setmOnePixelActivity(OnePixelActivity mOnePixelActivity) {
this.mOnePixelActivity = mOnePixelActivity;
}
/**
* 關(guān)閉 Activity 界面
*/
public void finishActivity(){
// 關(guān)閉 Activity 界面
this.mOnePixelActivity.finish();
// 不要長期持有該 Activity 界面
this.mOnePixelActivity = null;
}
}
5、AndroidManifest.xml 清單文件
主要是配置 1 像素 Activity 的親和性設(shè)置 , 不要把這個 Activity 放在與主 Activity 相同的任務(wù)棧中 , 否則在解除鎖定時 , 會拉起后臺的無關(guān)任務(wù)棧 ;
同時也要注意不要把 1 像素 Activity 展示到用戶眼前 , 對用戶透明即可 ;
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="kim.hsl.keep_progress_alive">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.Keep_Progress_Alive">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<!--
設(shè)置最近任務(wù)列表中不顯示該 Activity 組件 ( 不要被用戶察覺 )
android:excludeFromRecents="true"
設(shè)置 Activity 親和性
讓該界面在一個獨立的任務(wù)棧中 , 不要與本應(yīng)用的其它任務(wù)棧放在一起
避免解除鎖屏后 , 關(guān)閉 1 像素界面 , 將整個任務(wù)棧都喚醒
android:taskAffinity="kim.hsl.keep_progress_alive.alive"
-->
<activity
android:name=".OnePixelActivity"
android:theme="@style/OnePixelActivityTheme"
android:excludeFromRecents="true"
android:taskAffinity="kim.hsl.keep_progress_alive.onepixel" />
</application>
</manifest>
6、透明主題
要保證 1 像素 Activity 界面完全透明 ;
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Base application theme. -->
<style name="Theme.Keep_Progress_Alive" parent="Theme.MaterialComponents.DayNight.DarkActionBar">
<!-- Primary brand color. -->
<item name="colorPrimary">@color/purple_500</item>
<item name="colorPrimaryVariant">@color/purple_700</item>
<item name="colorOnPrimary">@color/white</item>
<!-- Secondary brand color. -->
<item name="colorSecondary">@color/teal_200</item>
<item name="colorSecondaryVariant">@color/teal_700</item>
<item name="colorOnSecondary">@color/black</item>
<!-- Status bar color. -->
<item name="android:statusBarColor" tools:targetApi="l">?attr/colorPrimaryVariant</item>
<!-- Customize your theme here. -->
</style>
<style name="OnePixelActivityTheme">
<!-- 清空窗口背景 -->
<item name="android:windowBackground">@null</item>
<!-- 設(shè)置窗口背景透明 -->
<item name="android:windowIsTranslucent">true</item>
</style>
</resources>
二、taskAffinity 親和性說明
<activity
android:name=".OnePixelActivity"
android:theme="@style/OnePixelActivityTheme"
android:excludeFromRecents="true"
android:taskAffinity="kim.hsl.keep_progress_alive.onepixel" />
Activity 在 AndroidManifest.xml 清單文件 中的 android:taskAffinity 親和性設(shè)置 , 主要是設(shè)置該 Activity 的任務(wù)棧 ;
親和性相同的 Activity 組件 , 放在同一個任務(wù)棧中 ;
應(yīng)用的親和性屬性默認就是包名 , 如果不設(shè)置 , 默認是在同一個任務(wù)棧中的 ;
① 親和性拉起 : 如果 Activity A 組件的 allowTaskReparenting 屬性設(shè)置為 true , 該 Activity 組件進入后臺 , 當有一個新的 Activity B 與 Activity A 組件有相同的親和性 , 那么 Activity A 會被拉起 , 放入 Activity B 所在的任務(wù)棧 ;
② 創(chuàng)建新的任務(wù)棧 : 啟動 Activity , 并且設(shè)置 Intent.FLAG_ACTIVITY_NEW_TASK 標志 , 會查詢是否有相同的親和性任務(wù)棧 , 如果有則將該 Activity 放入該任務(wù)棧 , 如果沒有 , 則創(chuàng)建一個新的任務(wù)棧 ; ( 本博客示例中 , 就使用了這種用法 )
③ 加載 SingleTask Activity : 首先檢查是否有相同親和性的 Activity , 如果有則銷毀已存在的 Activity 所在棧內(nèi)的 Activity 以上的界面 , 調(diào)用該 Activity 的 onNewIntent 方法 ; 如果沒有 , 則查詢是否有該任務(wù)棧 , 有任務(wù)棧便入棧 , 沒有任務(wù)棧就創(chuàng)建新的任務(wù)棧 ;
④ 加載 SingleInstance Activity : 首先檢查是否有相同親和性的 Activity ; 如果有則調(diào)用該 Activity 的 onNewIntent 方法 ; 如果沒有創(chuàng)建新的 Activity 放入新的任務(wù)棧 ;
三、測試
運行該應(yīng)用 , 獲取應(yīng)用的進程 PID = 3891 3891 3891 , 在 Android Studio 中查看即可 ;
查看日志發(fā)現(xiàn) , 廣播接收者已經(jīng)注冊 ;


查詢此時該應(yīng)用的 oom_adj 值為 0 0 0 , 前臺進程 ;

C:\Users\octop>adb shell walleye:/ $ su walleye:/ # cat /proc/3891/oom_adj 0 walleye:/ #
按下 Home 鍵 , 界面如下 , Logcat 日志基本沒有變化 ;

查詢該 PID 對應(yīng)的 oom_adj 值 12 12 12 , 后臺進程 ;

C:\Users\octop>adb shell walleye:/ $ su walleye:/ # cat /proc/3891/oom_adj 0 walleye:/ # cat /proc/3891/oom_adj 12 walleye:/ #
按下鎖屏鍵 , 查詢該 PID 對應(yīng)的 oom_adj 值 ,
界面鎖屏 ,

日志有更新 , 說明 1 像素 Activity 已經(jīng)啟動 ;

查詢該 PID 對應(yīng)的 oom_adj 值 0 0 0 , 前臺進程 ;

C:\Users\octop>adb shell walleye:/ $ su walleye:/ # cat /proc/3891/oom_adj 0 walleye:/ # cat /proc/3891/oom_adj 12 walleye:/ # cat /proc/3891/oom_adj 0 walleye:/ #
喚醒 , 查詢該 PID 對應(yīng)的 oom_adj 值 ,

日志信息中顯示 , 喚醒時 , 1 像素 Activity 退出 , 此時解除鎖屏 ;

查詢該 PID 對應(yīng)的 oom_adj 值 12 12 12 , 后臺進程 ;

C:\Users\octop>adb shell walleye:/ $ su walleye:/ # cat /proc/3891/oom_adj 0 walleye:/ # cat /proc/3891/oom_adj 12 walleye:/ # cat /proc/3891/oom_adj 0 walleye:/ # cat /proc/3891/oom_adj 12 walleye:/ #
該案例實現(xiàn)了在鎖屏?xí)r , 進程沒有被殺死 ;
以上就是Android進程?;钪嵘M程優(yōu)先級的詳細內(nèi)容,更多關(guān)于Android提升進程優(yōu)先級的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
一文詳解如何在Flutter中使用導(dǎo)航Navigator
一個APP如果沒有頁面跳轉(zhuǎn)那么是沒有靈魂的,頁面跳轉(zhuǎn)的一個常用說法就是Navigator。那么在flutter中如何使用Navigator呢?本文就來和大家詳細聊聊2023-02-02
Android中使用開源框架eventbus3.0實現(xiàn)fragment之間的通信交互
本文主要介紹了Android中使用開源框架eventbus3.0實現(xiàn)fragment之間的通信交互的方法,具有很好的參考價值,下面跟著小編一起來看下吧2017-02-02

