Android 开发中与Activity相关的几个小技巧

827 阅读6分钟

一、页面如何添加水印?

问题一是如何给页面添加水印?针对有一定安全性要求的App,通常需要在页面中添加水印。

在我们的项目中,我们是基于Activity添加水印。

大致代码可以为

FrameLayout waterMarkView = new FrameLayout(activity);
waterMarkView.setBackground(createWaterMarkDrawable(waterMarkText));
FrameLayout contentLayout = findViewById(android.R.id.content);
contentLayout.addView(waterMarkView, -1, -1);

每个Activity都有一个ContentView,由于ContentView是一个FrameLayout,那么我们就可以在不影响原生布局的情况下,再为其添加一个水印View。

二、如何统计Activity生命周期耗时

Android的开发同学都知道,在四大组件的生命周期的方法中是不可以执行耗时操作的。那么我们要怎么监控每个Activity的生命周期方法的耗时呢?

1、Android 10以上版本

在Android 10及以上我们可以考虑依赖 ActivityLifecycleCallbacks 接口,原因是在Android10以及以上,ActivityLifecycleCallbacks接口为每个生命周期的方法都新增了onActivityPreCreated(),onActivityPostCreated()配对方法。

public interface ActivityLifecycleCallbacks {

        /**
         * Called as the first step of the Activity being created. This is always called before
         * {@link Activity#onCreate}.
         */
        default void onActivityPreCreated(@NonNull Activity activity,
                @Nullable Bundle savedInstanceState) {
        }

        /**
         * Called when the Activity calls {@link Activity#onCreate super.onCreate()}.
         */
        void onActivityCreated(@NonNull Activity activity, @Nullable Bundle savedInstanceState);

        /**
         * Called as the last step of the Activity being created. This is always called after
         * {@link Activity#onCreate}.
         */
        default void onActivityPostCreated(@NonNull Activity activity,
                @Nullable Bundle savedInstanceState) {
        }

        /**
         * Called as the first step of the Activity being started. This is always called before
         * {@link Activity#onStart}.
         */
        default void onActivityPreStarted(@NonNull Activity activity) {
        }

        /**
         * Called when the Activity calls {@link Activity#onStart super.onStart()}.
         */
        void onActivityStarted(@NonNull Activity activity);

        /**
         * Called as the last step of the Activity being started. This is always called after
         * {@link Activity#onStart}.
         */
        default void onActivityPostStarted(@NonNull Activity activity) {
        }

        /**
         * Called as the first step of the Activity being resumed. This is always called before
         * {@link Activity#onResume}.
         */
        default void onActivityPreResumed(@NonNull Activity activity) {
        }

        /**
         * Called when the Activity calls {@link Activity#onResume super.onResume()}.
         */
        void onActivityResumed(@NonNull Activity activity);

        /**
         * Called as the last step of the Activity being resumed. This is always called after
         * {@link Activity#onResume} and {@link Activity#onPostResume}.
         */
        default void onActivityPostResumed(@NonNull Activity activity) {
        }

        /**
         * Called as the first step of the Activity being paused. This is always called before
         * {@link Activity#onPause}.
         */
        default void onActivityPrePaused(@NonNull Activity activity) {
        }

        /**
         * Called when the Activity calls {@link Activity#onPause super.onPause()}.
         */
        void onActivityPaused(@NonNull Activity activity);

        /**
         * Called as the last step of the Activity being paused. This is always called after
         * {@link Activity#onPause}.
         */
        default void onActivityPostPaused(@NonNull Activity activity) {
        }

        /**
         * Called as the first step of the Activity being stopped. This is always called before
         * {@link Activity#onStop}.
         */
        default void onActivityPreStopped(@NonNull Activity activity) {
        }

        /**
         * Called when the Activity calls {@link Activity#onStop super.onStop()}.
         */
        void onActivityStopped(@NonNull Activity activity);

        /**
         * Called as the last step of the Activity being stopped. This is always called after
         * {@link Activity#onStop}.
         */
        default void onActivityPostStopped(@NonNull Activity activity) {
        }

        /**
         * Called as the first step of the Activity saving its instance state. This is always
         * called before {@link Activity#onSaveInstanceState}.
         */
        default void onActivityPreSaveInstanceState(@NonNull Activity activity,
                @NonNull Bundle outState) {
        }

        /**
         * Called when the Activity calls
         * {@link Activity#onSaveInstanceState super.onSaveInstanceState()}.
         */
        void onActivitySaveInstanceState(@NonNull Activity activity, @NonNull Bundle outState);

        /**
         * Called as the last step of the Activity saving its instance state. This is always
         * called after{@link Activity#onSaveInstanceState}.
         */
        default void onActivityPostSaveInstanceState(@NonNull Activity activity,
                @NonNull Bundle outState) {
        }

        /**
         * Called as the first step of the Activity being destroyed. This is always called before
         * {@link Activity#onDestroy}.
         */
        default void onActivityPreDestroyed(@NonNull Activity activity) {
        }

        /**
         * Called when the Activity calls {@link Activity#onDestroy super.onDestroy()}.
         */
        void onActivityDestroyed(@NonNull Activity activity);

        /**
         * Called as the last step of the Activity being destroyed. This is always called after
         * {@link Activity#onDestroy}.
         */
        default void onActivityPostDestroyed(@NonNull Activity activity) {
        }

        /**
         * Called when the Activity configuration was changed.
         * @hide
         */
        default void onActivityConfigurationChanged(@NonNull Activity activity) {
        }
    }

统计耗时我们可以利用onActivityPostCreated()执行的时间减去onActivityPreCreated()执行的时间戳,就可以得出该onCreate的耗时时间了。

那么为什么onActivityPostCreated() 减去 onActivityPreCreated()的时间戳就可以呢?

我们可以简单看一下Activity performCreate 代码

    final void performCreate(Bundle icicle, PersistableBundle persistentState) {
        if (Trace.isTagEnabled(Trace.TRACE_TAG_WINDOW_MANAGER)) {
            Trace.traceBegin(Trace.TRACE_TAG_WINDOW_MANAGER, "performCreate:"
                    + mComponent.getClassName());
        }
        dispatchActivityPreCreated(icicle);
        mCanEnterPictureInPicture = true;
        // initialize mIsInMultiWindowMode and mIsInPictureInPictureMode before onCreate
        final int windowingMode = getResources().getConfiguration().windowConfiguration
                .getWindowingMode();
        mIsInMultiWindowMode = inMultiWindowMode(windowingMode);
        mIsInPictureInPictureMode = windowingMode == WINDOWING_MODE_PINNED;
        mShouldDockBigOverlays = getResources().getBoolean(R.bool.config_dockBigOverlayWindows);
        restoreHasCurrentPermissionRequest(icicle);
        if (persistentState != null) {
            onCreate(icicle, persistentState);
        } else {
            onCreate(icicle);
        }
        EventLogTags.writeWmOnCreateCalled(mIdent, getComponentName().getClassName(),
                "performCreate");
        mActivityTransitionState.readState(icicle);

        mVisibleFromClient = !mWindow.getWindowStyle().getBoolean(
                com.android.internal.R.styleable.Window_windowNoDisplay, false);
        mFragments.dispatchActivityCreated();
        mActivityTransitionState.setEnterActivityOptions(this, getActivityOptions());
        dispatchActivityPostCreated(icicle);
        Trace.traceEnd(Trace.TRACE_TAG_WINDOW_MANAGER);
    }

可以看到在performCreate中,先执行了dispatchActivityPreCreated()方法, 然后开始执行onCreate方法,而后执行dispatchActivityPostCreated()方法。

dispatchActivityPreCreated()

    private void dispatchActivityPreCreated(@Nullable Bundle savedInstanceState) {
        getApplication().dispatchActivityPreCreated(this, savedInstanceState);
        Object[] callbacks = collectActivityLifecycleCallbacks();
        if (callbacks != null) {
            for (int i = 0; i < callbacks.length; i++) {
                ((Application.ActivityLifecycleCallbacks) callbacks[i]).onActivityPreCreated(this,
                        savedInstanceState);
            }
        }
    }

dispatchActivityPostCreated()

    private void dispatchActivityPostCreated(@Nullable Bundle savedInstanceState) {
        Object[] callbacks = collectActivityLifecycleCallbacks();
        if (callbacks != null) {
            for (int i = 0; i < callbacks.length; i++) {
                ((Application.ActivityLifecycleCallbacks) callbacks[i]).onActivityPostCreated(this,
                        savedInstanceState);
            }
        }
        getApplication().dispatchActivityPostCreated(this, savedInstanceState);
    }

可以看到dispatchActivityPreCreated() 、dispatchActivityPostCreated()的作用就是通知Application以及Activity中注册的 ActivityLifecycleCallbacks。

以上简单介绍了一下onCreate方法,其他的生命周期方法也是同理,这里就不啰嗦了。

2、Android 9及以下版本监听生命周期方法耗时

当前Android9及以下版本在市场上占有量已经比较低了,由于低版本ActivityLifecycleCallbacks中没有可以利用的方法,所以在低版本中一般要使用Hook Instrumentation的方式。

如何Hook Instrumentation在社区也有成熟的代码,这里简单贴一下:

private static void hookInstrumentation() throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, NoSuchFieldException {

        Class<?> c = Class.forName("android.app.ActivityThread");
        Method currentActivityThread = c.getDeclaredMethod("currentActivityThread");
        boolean acc = currentActivityThread.isAccessible();
        if (!acc) {
            currentActivityThread.setAccessible(true);
        }
        //获取ActivityThread对象
        Object o = currentActivityThread.invoke(null);

        Field f = c.getDeclaredField("mInstrumentation");
        acc = f.isAccessible();
        if (!acc) {
            f.setAccessible(true);
        }
        //获取Instrumentation对象
        Instrumentation currentInstrumentation = (Instrumentation) f.get(o);
        Instrumentation ins = new CustomInstrumentation(currentInstrumentation);
        //替换
        f.set(o, ins);
        if (!acc) {
            f.setAccessible(acc);
        }
    }

这里 CustomInstrumentation 是自定义的Instrumentation子类。

在CustomInstrumentation中重写 callActivityOnCreate() callActivityOnStart()等方法,就可以在方法内部添加计算耗时逻辑。

三、如何获取应用中Top Activity

在平时工作中,通常需要获取App的Top Activity 来协助定位逻辑,这里介绍几种方式:

1、通过ADB

通过ADB命令查看任意App的Top Activity:

adb shell "dumpsys window | grep mFocusedApp"

2、利用ActivityManager

利用ActivityManager获取 ActivityManager.RunningTaskInfo 对象,其内有很多数据我们可以应用,例如获取TopActivity的名称:

    fun getTopActivityName() {
        val activityManager: ActivityManager =
            getSystemService(ACTIVITY_SERVICE) as ActivityManager
        val tasks: List<ActivityManager.RunningTaskInfo> = activityManager.getRunningTasks(1)
        if (tasks[0].topActivity != null) {
            val text = tasks[0].topActivity?.packageName 
            ...
        }
    }

我们团队内部有开发一个开发者工具,支持利用浮窗实时显示当前Top Activity的名称,以此提高大家定位Activity的效率,我们使用的是这个逻辑。

四、如何判断App是否在前台?

在我们的App中,我们通常需要监听App前后台的状态切换。例如当App切换到前台时,会立即检查长连接是否存在,不过不存在会立即去建立长连接。

这里判断App是否在前台也是通过Activity来判断。

1、依然是通过生命周期回调

方案一依然是通过 ActivityLifecycleCallbacks 接口来自行维护(不得不说这个接口非常有用),通过计数的方式在内存维护前台Activity的个数。当个数为0时,则认为App切换到了后台。

不过在实践过程中我们发现这个逻辑存在偶现的Bug,由于是偶现产生的Bug的原因我们没有实际上搞清楚,能够确定的是大概率与系统有关。基于此引入方案二。

2、通过ActivityManager

在高版本中利用 ActivityManager 获取到RunningTaskInfo对象之后,通过isVisible()方法来判断。

    public void checkAppBackgroundImpl() {
        if (Build.VERSION.SDK_INT > Build.VERSION_CODES.TIRAMISU) {
            ActivityManager activityManager = (ActivityManager) App.getContext().getSystemService(Context.ACTIVITY_SERVICE);
            List<ActivityManager.RunningTaskInfo> tasks = activityManager.getRunningTasks(1);
            if (tasks.get(0) != null) {
                ActivityManager.RunningTaskInfo runningTaskInfo = tasks.get(0);
                boolean visible = runningTaskInfo.isVisible();
                ...
            }
        }
    }

在实践过程中,这个方案还没有被反馈遇到Bug,相对来讲比较准确,确定就是需要高版本。

五、总结

本篇主要是介绍几个与Activity相关的小技巧,都是在工作中可以经常使用的,希望能够读者带来一些启发。