Android進(jìn)程?;钪嵘M(jìn)程優(yōu)先級
一、1 像素 Activity 提高進(jìn)程優(yōu)先級
使用 Activity 可以提升進(jìn)程的 oom_adj 值 ;
APP 進(jìn)入后臺后 , 使用 BroadcastReceiver 廣播接收者 , 監(jiān)聽 Android 系統(tǒng)的鎖屏廣播事件 ;
- 屏幕鎖定 : 啟動只有 1 1 1 像素的透明 Activity 界面 ;
- 屏幕解鎖 : 退出上述 1 1 1 像素的透明 Activity 界面 ;
1、主界面 MainActivity
主界面 , 主要負(fù)責(zé)注冊廣播接收者 ;
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 , 有可能有進(jìn)程保活的同行 , 也彈出個同樣類型的 Activity , 一般都是透明的 , 即使這樣 , 最次也是個可見進(jìn)程 ;
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) 坐標(biāo)開始放置 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、管理類
單例管理類 , 負(fù)責(zé)注冊廣播接收者 , 在廣播接收者中啟動 1 像素頁面 , 同時也負(fù)責(zé)關(guān)閉該 1 像素頁面 ;
該管理類負(fù)責(zé) 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)用的親和性屬性默認(rèn)就是包名 , 如果不設(shè)置 , 默認(rèn)是在同一個任務(wù)棧中的 ;
① 親和性拉起 : 如果 Activity A 組件的 allowTaskReparenting 屬性設(shè)置為 true , 該 Activity 組件進(jìn)入后臺 , 當(dāng)有一個新的 Activity B 與 Activity A 組件有相同的親和性 , 那么 Activity A 會被拉起 , 放入 Activity B 所在的任務(wù)棧 ;
② 創(chuàng)建新的任務(wù)棧 : 啟動 Activity , 并且設(shè)置 Intent.FLAG_ACTIVITY_NEW_TASK 標(biāo)志 , 會查詢是否有相同的親和性任務(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)用的進(jìn)程 PID = 3891 3891 3891 , 在 Android Studio 中查看即可 ;
查看日志發(fā)現(xiàn) , 廣播接收者已經(jīng)注冊 ;
查詢此時該應(yīng)用的 oom_adj 值為 0 0 0 , 前臺進(jìn)程 ;
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 , 后臺進(jìn)程 ;
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 , 前臺進(jìn)程 ;
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 , 后臺進(jìn)程 ;
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 , 進(jìn)程沒有被殺死 ;
以上就是Android進(jìn)程?;钪嵘M(jìn)程優(yōu)先級的詳細(xì)內(nèi)容,更多關(guān)于Android提升進(jìn)程優(yōu)先級的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Android中監(jiān)聽系統(tǒng)網(wǎng)絡(luò)連接打開或者關(guān)閉的實現(xiàn)代碼
本篇文章對Android中監(jiān)聽系統(tǒng)網(wǎng)絡(luò)連接打開或者關(guān)閉的實現(xiàn)用實例進(jìn)行了介紹。需要的朋友參考下2013-05-05Windows下React Native的Android環(huán)境部署及布局示例
這篇文章主要介紹了Windows下React Native的Android環(huán)境部署及布局示例,這里安卓開發(fā)IDE建議使用Android Studio,且為Windows安裝npm包管理器,需要的朋友可以參考下2016-03-03Android編程自定義title bar(標(biāo)題欄)示例
這篇文章主要介紹了Android編程自定義title bar(標(biāo)題欄)的方法,結(jié)合實例形式分析了Android針對標(biāo)題欄的設(shè)置與頁面布局操作相關(guān)技巧,需要的朋友可以參考下2016-10-10Android SurfaceView運行機(jī)制剖析--處理切換到后臺再重新進(jìn)入程序時的異常
本文主要介紹Android SurfaceView運行機(jī)制,這里整理了詳細(xì)的資料來講解SurfaceView的運行原理,并附示例代碼參考,有需要的小伙伴可以參考下2016-08-08Android開發(fā)必備知識 為什么說Kotlin值得一試
為什么說值得一試,這篇文章主要為大家詳細(xì)介紹了Android開發(fā)必備知識,Kotlin的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-05-05