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
在鎖屏時 , 彈出的 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 設置成 1 像素 WindowManager.LayoutParams layoutParams = window.getAttributes(); // 寬高都設置 1 像素 layoutParams.width = 1; layoutParams.height = 1; // 放置位置 (0, 0) 坐標開始放置 layoutParams.x = 0; layoutParams.y = 0; // 在將布局參數(shù)設置會 Window 對象中 window.setAttributes(layoutParams); // 設置界面到 KeepProgressAliveManager 單例對象中 , 用于關閉界面 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 , 兩個廣播 , 再鎖屏時啟動 1 像素 Activity , 在解除鎖屏時 , 關閉 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)){ // 鎖屏時開啟 Activity KeepProgressAliveManager.getmInstance().startActivity(context); Log.i("KeepProgressAliveReceiver", "鎖屏, 開啟 1 像素 Activity"); }else if (Intent.ACTION_SCREEN_ON.equals(action)){ // 解除所屏蔽關閉 Activity KeepProgressAliveManager.getmInstance().finishActivity(); Log.i("KeepProgressAliveReceiver", "解除鎖屏, 關閉 1 像素 Activity"); } } }
4、管理類
單例管理類 , 負責注冊廣播接收者 , 在廣播接收者中啟動 1 像素頁面 , 同時也負責關閉該 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)建一個任務棧 ( 前提是親和性不同 ) intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); context.startActivity(intent); } /** * 設置 1 像素 Activity 界面, 用于關閉 * @param mOnePixelActivity */ public void setmOnePixelActivity(OnePixelActivity mOnePixelActivity) { this.mOnePixelActivity = mOnePixelActivity; } /** * 關閉 Activity 界面 */ public void finishActivity(){ // 關閉 Activity 界面 this.mOnePixelActivity.finish(); // 不要長期持有該 Activity 界面 this.mOnePixelActivity = null; } }
5、AndroidManifest.xml 清單文件
主要是配置 1 像素 Activity 的親和性設置 , 不要把這個 Activity 放在與主 Activity 相同的任務棧中 , 否則在解除鎖定時 , 會拉起后臺的無關任務棧 ;
同時也要注意不要把 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> <!-- 設置最近任務列表中不顯示該 Activity 組件 ( 不要被用戶察覺 ) android:excludeFromRecents="true" 設置 Activity 親和性 讓該界面在一個獨立的任務棧中 , 不要與本應用的其它任務棧放在一起 避免解除鎖屏后 , 關閉 1 像素界面 , 將整個任務棧都喚醒 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> <!-- 設置窗口背景透明 --> <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 親和性設置 , 主要是設置該 Activity 的任務棧 ;
親和性相同的 Activity 組件 , 放在同一個任務棧中 ;
應用的親和性屬性默認就是包名 , 如果不設置 , 默認是在同一個任務棧中的 ;
① 親和性拉起 : 如果 Activity A 組件的 allowTaskReparenting 屬性設置為 true , 該 Activity 組件進入后臺 , 當有一個新的 Activity B 與 Activity A 組件有相同的親和性 , 那么 Activity A 會被拉起 , 放入 Activity B 所在的任務棧 ;
② 創(chuàng)建新的任務棧 : 啟動 Activity , 并且設置 Intent.FLAG_ACTIVITY_NEW_TASK 標志 , 會查詢是否有相同的親和性任務棧 , 如果有則將該 Activity 放入該任務棧 , 如果沒有 , 則創(chuàng)建一個新的任務棧 ; ( 本博客示例中 , 就使用了這種用法 )
③ 加載 SingleTask Activity : 首先檢查是否有相同親和性的 Activity , 如果有則銷毀已存在的 Activity 所在棧內的 Activity 以上的界面 , 調用該 Activity 的 onNewIntent 方法 ; 如果沒有 , 則查詢是否有該任務棧 , 有任務棧便入棧 , 沒有任務棧就創(chuàng)建新的任務棧 ;
④ 加載 SingleInstance Activity : 首先檢查是否有相同親和性的 Activity ; 如果有則調用該 Activity 的 onNewIntent 方法 ; 如果沒有創(chuàng)建新的 Activity 放入新的任務棧 ;
三、測試
運行該應用 , 獲取應用的進程 PID = 3891 3891 3891 , 在 Android Studio 中查看即可 ;
查看日志發(fā)現(xiàn) , 廣播接收者已經(jīng)注冊 ;
查詢此時該應用的 oom_adj 值為 0 0 0 , 前臺進程 ;
C:\Users\octop>adb shell walleye:/ $ su walleye:/ # cat /proc/3891/oom_adj 0 walleye:/ #
按下 Home 鍵 , 界面如下 , Logcat 日志基本沒有變化 ;
查詢該 PID 對應的 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 對應的 oom_adj 值 ,
界面鎖屏 ,
日志有更新 , 說明 1 像素 Activity 已經(jīng)啟動 ;
查詢該 PID 對應的 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 對應的 oom_adj 值 ,
日志信息中顯示 , 喚醒時 , 1 像素 Activity 退出 , 此時解除鎖屏 ;
查詢該 PID 對應的 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)了在鎖屏時 , 進程沒有被殺死 ;
以上就是Android進程?;钪嵘M程優(yōu)先級的詳細內容,更多關于Android提升進程優(yōu)先級的資料請關注腳本之家其它相關文章!
相關文章
Android中監(jiān)聽系統(tǒng)網(wǎng)絡連接打開或者關閉的實現(xiàn)代碼
本篇文章對Android中監(jiān)聽系統(tǒng)網(wǎng)絡連接打開或者關閉的實現(xiàn)用實例進行了介紹。需要的朋友參考下2013-05-05Windows下React Native的Android環(huán)境部署及布局示例
這篇文章主要介紹了Windows下React Native的Android環(huán)境部署及布局示例,這里安卓開發(fā)IDE建議使用Android Studio,且為Windows安裝npm包管理器,需要的朋友可以參考下2016-03-03Android SurfaceView運行機制剖析--處理切換到后臺再重新進入程序時的異常
本文主要介紹Android SurfaceView運行機制,這里整理了詳細的資料來講解SurfaceView的運行原理,并附示例代碼參考,有需要的小伙伴可以參考下2016-08-08Android開發(fā)必備知識 為什么說Kotlin值得一試
為什么說值得一試,這篇文章主要為大家詳細介紹了Android開發(fā)必備知識,Kotlin的相關資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-05-05