安卓在Recent界面清理app同时也清理掉关于此app的所有服务

在此网站上冲浪的大佬基本都是做安卓的,什么是服务我就不多赘述了,在安卓系统里有很多服务在后台运行,不管是app的还是系统的都有,比较重要的systemservice,有些app在多任务后台清理掉以后一些服务还在后台一直运行,比如QQ音乐,后台清理掉,后台播放着音乐一样不会停止掉,这样会导致占用系统资源,时间久了后台运行的服务多了,系统就会变卡,但是原生的系统是不会管这个的,所以需要厂商自己客制,最近一个客户也是要做这个功能,顾客是上帝(客户纯byd,天天提抽象又难做的需求,公司又不加工资,纯牛马),没办法只能给他做了,顺便也写一篇文章(好久在这上面写了)。 我会把低版本和高版本的方法都会写出来,主要是安卓8.1,安卓11和安卓十三,中间的版本需要改的也是大同小异,主要挑差异大的写,要想清理服务,需要借助ActivityManager.java里面的forceStopPackage方法,因为我在安卓八点一、安卓七改的地方context是没有定义的,我自己加的context不能用,用ActivityManager编译一直报错,怎么改都有问题,所以我用了IActivityManager.aidl里面的forceStopPackage接口方法,下面先看一下安卓八点一的forceStopPackage方法源码:
void forceStopPackage(in String packageName, int userId);
因为IActivityManager.aidl是一个接口类,实现是在ActivityManager.java,所以还是要看ActivityManager.java里面的forceStopPackage方法:
/**
     * @see #forceStopPackageAsUser(String, int)
     * @hide
     */
    @SystemApi
    @RequiresPermission(Manifest.permission.FORCE_STOP_PACKAGES)
    public void forceStopPackage(String packageName) {
        forceStopPackageAsUser(packageName, UserHandle.myUserId());
    }
forceStopPackage方法在各个版本的写法基本相同的,我就不贴出来了,作用就是用于强行停止指定应用包方法,注意点是要注意在AndroidManifest里面加一个FORCE_STOP_PACKAGES权限,不然就会报错。 知道了怎么杀服务的方法以后,下面就要怎么实现在recent界面划掉app的时候加上这个方法,在recent界面划掉app的时候会调用一个方法,这个方法是removeTask,这个方法在安卓九以下是在systemui里面,因为从安卓九开始,recent的一些功能被放在launcher3里面的quickstep,首先看安卓八点一,安卓八点一的removeTask方法是在frameworks/base/packages/SystemUI/src/com/android/systemui/recents/model/TaskStack.java里面
    /**
     * Removes a task from the stack, with an additional {@param animation} hint to the callbacks on
     * how they should update themselves.
     */
    public void removeTask(Task t, AnimationProps animation, boolean fromDockGesture,
            boolean dismissRecentsIfAllRemoved) {
        if (mStackTaskList.contains(t)) {
            removeTaskImpl(mStackTaskList, t);
            Task newFrontMostTask = getStackFrontMostTask(false  /* includeFreeform */);
            if (mCb != null) {
                // Notify that a task has been removed
                mCb.onStackTaskRemoved(this, t, newFrontMostTask, animation,
                        fromDockGesture, dismissRecentsIfAllRemoved);
            }
        }
        mRawTaskList.remove(t);
    }
    /**
     * Removes all tasks from the stack.
     */
    public void removeAllTasks(boolean notifyStackChanges) {
        ArrayList<Task> tasks = mStackTaskList.getTasks();
        for (int i = tasks.size() - 1; i >= 0; i--) {
            Task t = tasks.get(i);
            removeTaskImpl(mStackTaskList, t);
            mRawTaskList.remove(t);
        }
        if (mCb != null && notifyStackChanges) {
            // Notify that all tasks have been removed
            mCb.onStackTasksRemoved(this);
        }
    }
单个划掉app用的是removeTask,全部清理是removeAllTasks,下面直接看解决办法:
diff --git a/frameworks/base/packages/SystemUI/AndroidManifest.xml b/frameworks/base/packages/SystemUI/AndroidManifest.xml
index 2f7f0bb..6e83978 100755
--- a/frameworks/base/packages/SystemUI/AndroidManifest.xml
+++ b/frameworks/base/packages/SystemUI/AndroidManifest.xml
@@ -33,6 +33,7 @@
     <!-- Used to read storage for all users -->
     <uses-permission android:name="android.permission.WRITE_MEDIA_STORAGE" />
     <uses-permission android:name="android.permission.WAKE_LOCK" />
+    <uses-permission android:name="android.permission.FORCE_STOP_PACKAGES"/>
 
     <uses-permission android:name="android.permission.INJECT_EVENTS" />
     <uses-permission android:name="android.permission.DUMP" />
diff --git a/frameworks/base/packages/SystemUI/src/com/android/systemui/recents/model/TaskStack.java b/frameworks/base/packages/SystemUI/src/com/android/systemui/recents/model/TaskStack.java
old mode 100644
new mode 100755
index 6e3be09..f63c785
--- a/frameworks/base/packages/SystemUI/src/com/android/systemui/recents/model/TaskStack.java
+++ b/frameworks/base/packages/SystemUI/src/com/android/systemui/recents/model/TaskStack.java
@@ -642,6 +642,14 @@ public class TaskStack {
     public void removeTask(Task t, AnimationProps animation, boolean fromDockGesture,
             boolean dismissRecentsIfAllRemoved) {
         if (mStackTaskList.contains(t)) {
+            IActivityManager activityManager = ActivityManagerNative.getDefault();
+            String pkgName = t.key.getComponent().getPackageName();
+            Log.d("swl", "Stopping pkgName " + pkgName);
+            try {
+                activityManager.forceStopPackage(pkgName, UserHandle.myUserId());
+            } catch (RemoteException e) {
+                Log.e("swl", "Failed to force stop package " + pkgName, e);
+            }
             removeTaskImpl(mStackTaskList, t);
             Task newFrontMostTask = getStackFrontMostTask(false  /* includeFreeform */);
             if (mCb != null) {
@@ -658,8 +666,16 @@ public class TaskStack {
      */
     public void removeAllTasks(boolean notifyStackChanges) {
         ArrayList<Task> tasks = mStackTaskList.getTasks();
+        IActivityManager activityManager = ActivityManagerNative.getDefault();
         for (int i = tasks.size() - 1; i >= 0; i--) {
             Task t = tasks.get(i);
+            String pkgName = t.key.getComponent().getPackageName();
+            Log.d("swl", "Stopping allpkgName " + pkgName);  
+            try {
+                activityManager.forceStopPackage(pkgName, UserHandle.myUserId());
+            } catch (RemoteException e) {
+                Log.e("swl", "Failed to force stop allpackage " + pkgName, e);
+            }
             removeTaskImpl(mStackTaskList, t);
             mRawTaskList.remove(t);
         }
这是安卓八点一的方法,下面看安卓十一的方法,安卓十一的removeTask方法是在packages\apps\Launcher3\quickstep\recents_ui_overrides\src\com\android\quickstep\views\RecentsView.java中,源码如下:
private void removeTask(TaskView taskView, int index, EndState endState) {
        if (taskView.getTask() != null) {
            ActivityManagerWrapper.getInstance().removeTask(taskView.getTask().key.id);
            ComponentKey compKey = TaskUtils.getLaunchComponentKeyForTask(taskView.getTask().key);
            mActivity.getUserEventDispatcher().logTaskLaunchOrDismiss(
                    endState.logAction, Direction.UP, index, compKey);
            mActivity.getStatsLogManager().logger().withItemInfo(taskView.getItemInfo())
                    .log(LAUNCHER_TASK_DISMISS_SWIPE_UP);
        }
    }
具体的改法我就不写了,这里可以直接使用ActivityManager.java里面的forceStopPackage方法,此方案在(9-11都可适用)。 还有安卓十三的方法:
diff --git a/frameworks/base/data/etc/com.android.launcher3.xml b/frameworks/base/data/etc/com.android.launcher3.xml
old mode 100644
new mode 100755
index 36a51341f52..9d6b004ccb6
--- a/frameworks/base/data/etc/com.android.launcher3.xml
+++ b/frameworks/base/data/etc/com.android.launcher3.xml
@@ -25,5 +25,6 @@
         <permission name="android.permission.START_TASKS_FROM_RECENTS"/>
         <permission name="android.permission.STATUS_BAR"/>
         <permission name="android.permission.STOP_APP_SWITCHES"/>
+               <permission name="android.permission.FORCE_STOP_PACKAGES"/>
     </privapp-permissions>
 </permissions>
diff --git a/vendor/mediatek/proprietary/packages/apps/Launcher3/quickstep/AndroidManifest.xml b/vendor/mediatek/proprietary/packages/apps/Launcher3/quickstep/AndroidManifest.xml
old mode 100644
new mode 100755
index 352cd3e7b62..1267682a5e3
--- a/vendor/mediatek/proprietary/packages/apps/Launcher3/quickstep/AndroidManifest.xml
+++ b/vendor/mediatek/proprietary/packages/apps/Launcher3/quickstep/AndroidManifest.xml
@@ -37,6 +37,7 @@
     <uses-permission android:name="android.permission.MANAGE_ACCESSIBILITY"/>
     <uses-permission android:name="android.permission.MONITOR_INPUT"/>
     <uses-permission android:name="android.permission.ALLOW_SLIPPERY_TOUCHES"/>
+    <uses-permission android:name="android.permission.FORCE_STOP_PACKAGES"/>
 
     <uses-permission android:name="android.permission.SYSTEM_APPLICATION_OVERLAY" />
 
diff --git a/vendor/mediatek/proprietary/packages/apps/Launcher3/quickstep/src/com/android/quickstep/views/RecentsView.java b/vendor/mediatek/proprietary/packages/apps/Launcher3/quickstep/src/com/android/quickstep/views/RecentsView.java
old mode 100644
new mode 100755
index 2360396cf70..d2d260c64de
--- a/vendor/mediatek/proprietary/packages/apps/Launcher3/quickstep/src/com/android/quickstep/views/RecentsView.java
+++ b/vendor/mediatek/proprietary/packages/apps/Launcher3/quickstep/src/com/android/quickstep/views/RecentsView.java
@@ -189,6 +189,9 @@ import java.util.List;
 import java.util.Objects;
 import java.util.function.Consumer;
 
+import android.app.ActivityManager;
+import android.app.ActivityOptions;
+
 /**
  * A list of recent tasks.
  */
@@ -3127,12 +3130,16 @@ public abstract class RecentsView<ACTIVITY_TYPE extends StatefulActivity<STATE_T
                if (success) {
                    if (shouldRemoveTask) {
                        if (dismissedTaskView.getTask() != null) {
+                       tring pkgname = dismissedTaskView.getTask().key.getPackageName();
+                       Log.d("swl","pkgname: "+pkgname);
                        if (ENABLE_QUICKSTEP_LIVE_TILE.get()&& dismissedTaskView.isRunningTask()) {
                            finishRecentsAnimation(true /* toRecents */, false /* shouldPip */,
-                           () -> removeTaskInternal(dismissedTaskViewId));
+                           () -> removeTaskInternal(dismissedTaskViewId,pkgname));
                        } else {
-                           removeTaskInternal(dismissedTaskViewId);
+                           removeTaskInternal(dismissedTaskViewId,pkgname);
                             }
                             mActivity.getStatsLogManager().logger()
                                     .withItemInfo(dismissedTaskView.getItemInfo())
@@ -3401,7 +3408,7 @@ public abstract class RecentsView<ACTIVITY_TYPE extends StatefulActivity<STATE_T
         return lastVisibleIndex;
     }
 
-    private void removeTaskInternal(int dismissedTaskViewId) {
+    private void removeTaskInternal(int dismissedTaskViewId, String pkgname) {
         int[] taskIds = getTaskIdsForTaskViewId(dismissedTaskViewId);
         int primaryTaskId = taskIds[0];
         int secondaryTaskId = taskIds[1];
@@ -3411,6 +3418,10 @@ public abstract class RecentsView<ACTIVITY_TYPE extends StatefulActivity<STATE_T
                     if (secondaryTaskId != -1) {
                         ActivityManagerWrapper.getInstance().removeTask(secondaryTaskId);
                     }
+                                       ActivityManager am = (ActivityManager) mActivity.getSystemService(Context.ACTIVITY_SERVICE);
+                                       Log.d("swl","pkgname: "+pkgname);
+                                       am.forceStopPackage(pkgname);                                   
                 },
                 REMOVE_TASK_WAIT_FOR_APP_STOP_MS);
     }
@@ -3436,6 +3447,12 @@ public abstract class RecentsView<ACTIVITY_TYPE extends StatefulActivity<STATE_T
         int count = getTaskViewCount();
         for (int i = 0; i < count; i++) {
             addDismissedTaskAnimations(requireTaskViewAt(i), duration, anim);
+                       //andy
+                       String pkgname = requireTaskViewAt(i).getTask().key.getPackageName();
+                       Log.d("swl","forceStop: "+pkgname););
+                       ActivityManager am = (ActivityManager) mActivity.getSystemService(Context.ACTIVITY_SERVICE);
+                       am.forceStopPackage(pkgname);                                   
+                       
         }
 
         mPendingAnimation = anim;
核心思路基本都是一样的,只是改的方式有点差异,我这个也是参考了一个MTK网站上几年前一个同事提的安卓九的case改出来的,安卓CV工程师哈哈哈,有大佬还有更好的方式,欢迎评论留言





























