Android進(jìn)程保活之提升進(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'> <applicationandroid: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:Usersoctop>adb shellwalleye:/ $ suwalleye:/ # cat /proc/3891/oom_adj0walleye:/ #
按下 Home 鍵 , 界面如下 , Logcat 日志基本沒有變化 ;
查詢該 PID 對應(yīng)的 oom_adj 值 12 12 12 , 后臺進(jìn)程 ;
C:Usersoctop>adb shellwalleye:/ $ suwalleye:/ # cat /proc/3891/oom_adj0walleye:/ # cat /proc/3891/oom_adj12walleye:/ #
按下鎖屏鍵 , 查詢該 PID 對應(yīng)的 oom_adj 值 ,
界面鎖屏 ,
日志有更新 , 說明 1 像素 Activity 已經(jīng)啟動 ;
查詢該 PID 對應(yīng)的 oom_adj 值 0 0 0 , 前臺進(jìn)程 ;
C:Usersoctop>adb shellwalleye:/ $ suwalleye:/ # cat /proc/3891/oom_adj0walleye:/ # cat /proc/3891/oom_adj12walleye:/ # cat /proc/3891/oom_adj0walleye:/ #
喚醒 , 查詢該 PID 對應(yīng)的 oom_adj 值 ,
日志信息中顯示 , 喚醒時 , 1 像素 Activity 退出 , 此時解除鎖屏 ;
查詢該 PID 對應(yīng)的 oom_adj 值 12 12 12 , 后臺進(jìn)程 ;
C:Usersoctop>adb shellwalleye:/ $ suwalleye:/ # cat /proc/3891/oom_adj0walleye:/ # cat /proc/3891/oom_adj12walleye:/ # cat /proc/3891/oom_adj0walleye:/ # cat /proc/3891/oom_adj12walleye:/ #
該案例實現(xiàn)了在鎖屏?xí)r , 進(jìn)程沒有被殺死 ;
以上就是Android進(jìn)程保活之提升進(jìn)程優(yōu)先級的詳細(xì)內(nèi)容,更多關(guān)于Android提升進(jìn)程優(yōu)先級的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. ASP 信息提示函數(shù)并作返回或者轉(zhuǎn)向2. asp(vbs)Rs.Open和Conn.Execute的詳解和區(qū)別及&H0001的說明3. CSS hack用法案例詳解4. PHP設(shè)計模式中工廠模式深入詳解5. 用css截取字符的幾種方法詳解(css排版隱藏溢出文本)6. ASP+ajax實現(xiàn)頂一下、踩一下同支持與反對的實現(xiàn)代碼7. JSP數(shù)據(jù)交互實現(xiàn)過程解析8. ThinkPHP5實現(xiàn)JWT Token認(rèn)證的過程(親測可用)9. asp中response.write("中文")或者js中文亂碼問題10. PHP session反序列化漏洞超詳細(xì)講解
