探究 Android 中的 ActivityLifecycleCallbacks

我一定会爱上你 - 谢春花

我一定会爱上你

ActivityLifecycleCallbacks 是用来监听所有 Activity 的生命周期回调。接口定义如下:

public interface ActivityLifecycleCallbacks {
    void onActivityCreated(Activity activity, Bundle savedInstanceState);
    void onActivityStarted(Activity activity);
    void onActivityResumed(Activity activity);
    void onActivityPaused(Activity activity);
    void onActivityStopped(Activity activity);
    void onActivitySaveInstanceState(Activity activity, Bundle outState);
    void onActivityDestroyed(Activity activity);
}

Activity 的每一个生命周期都对应 ActivityLifecycleCallbacks 接口中的一个方法,比如 onActivityCreated 回调是在 Activity 的 onCreate 方法中调用 getApplication().dispatchActivityCreated(this, savedInstanceState) 完成对 Activity 生命周期跟踪监听。

ActivityLifecycleCallbacks 使用

  • 要求 API 14+
package com.sunpeng.lifecycle;

import android.app.Application;

/**
 * Created by sunpeng on 17/6/2.
 */
public class MainApplication extends Application {

    @Override
    public void onCreate() {
        super.onCreate();
        // AppLifecycleCallback 实现 ActivityLifecycleCallbacks 接口方法
        this.registerActivityLifecycleCallbacks(new AppLifecycleCallback());
    }
}

探究在 Android 中的应用

  • 应用新开进程假重启处理(低内存回收、修改权限)
  • 管理 Activity 页面栈
  • 获取当前 Activity 页面
  • 判断应用前后台
  • 保存恢复状态值 savedInstanceState
  • 页面分析统计埋点

结合 ActivityLifecycleCallbacks ,实现了以上的一些功能,完整源码可以查看AppLifecycleCallback

1、应用新开进程假重启处理(低内存回收、修改权限)

应用在低内存的情况下退出重新启动,并不会执行正常的启动流程,而是创建新的进程,直接还原上一次的操作页面。比如:

  • 当前操作页面:LoginActivity
  • 正常启动使用流程:SplashActivity -> MainActivity -> LoginActivity
  • 低内存重启流程:** 新开进程,直接启动 LoginActivity **

** 低内存重启流程存在的问题:**页面栈信息丢失,页面显示以及返回跳转异常;MainActivity 可能没有执行,部分功能不会初始化。

** 解决思路:** 通过监听回调方法 onActivityCreated,判断应用启动的第一个 Activity 页面是否为 LauncherActivity,如果不是,则强制启动 LauncherActivity 来执行正常的启动流程。

// AppLifecycleCallback.java

private static final int APP_STATUS_UNKNOWN = -1;
private static final int APP_STATUS_LIVE = 0;
private int appStatus = APP_STATUS_UNKNOWN;

@Override
public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
    if (appStatus == APP_STATUS_UNKNOWN) {
        // 设置状态标志 APP_STATUS_LIVE
        appStatus = APP_STATUS_LIVE;
        // 判断是否跳转闪屏页
        startLauncherActivity(activity);
    }
}

private static void startLauncherActivity(Activity activity) {
    try {
        Intent launchIntent = activity.getPackageManager().getLaunchIntentForPackage(activity.getPackageName());
        String launcherClassName = launchIntent.getComponent().getClassName();
        String className = activity.getComponentName().getClassName();

        if (TextUtils.isEmpty(launcherClassName) || launcherClassName.equals(className)) {
            return;
        }

        Log.e(TAG, "launcher ClassName --> " + launcherClassName);
        Log.e(TAG, "current ClassName --> " + className);

        launchIntent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
        activity.startActivity(launchIntent);
        activity.finish();
    } catch (Exception e) {
        e.printStackTrace();
    }
}

** 验证方式:**
1、 通过 Android Studio 工具 Android Device Monitor 模拟杀死当前应用进程。(Tools -> Android -> Android Device Monitor

Android Device Monitor

2、App 退到后台,修改应用权限(6.0 以上系统),再次 App 回到前台,同样会出现应用新开进程重启。

2、管理 Activity 页面栈

Activity 页面栈,最常用的实现就是用来完全退出应用。ActivityLifecycleCallbacksStack 来管理所有的Activity,不仅方便集中管理存储 Activity 实例,也不容易造成内存泄露。

监听回调方法 onActivityCreatedonActivityDestroyed 添加删除 Actvity 实例。

// AppLifecycleCallback.java

@Override
public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
    ActivityStackManager.getInstance().addActivity(activity);
}

@Override
public void onActivityDestroyed(Activity activity) {
    ActivityStackManager.getInstance().removeActivity(activity);
}
// ActivityStackManager.java

public void addActivity(Activity activity){
    if (activities == null) {
        activities = new Stack<Activity>();
    }
    if (activities.search(activity) == -1) {
        activities.push(activity);
    }
}

public void removeActivity(Activity activity){
    if (activities != null && activities.size()>0){
        activities.remove(activity);
    }
}

3、获取当前 Activity 页面

React Native 开发,我们经常需要获取当前的 TopActivity 实例,用来弹出安全键盘、RN接口通信等。关于获取当前Activity的一些思考详细介绍了一些思路,结合具体实现,推荐两个实现思路:** 弱引用持有当前 Activity 实例和 Activity 页面栈方式。**

通过监听回调方法 onActivityResumed,设置当前 Activity 页面

// AppLifecycleCallback.java

@Override
public void onActivityResumed(Activity activity) {
    // 弱引用持有当前 Activity 实例
    ActivityStackManager.getInstance().setCurrentActivity(activity);
    // Activity 页面栈方式
    ActivityStackManager.getInstance().setTopActivity(activity);
}

** 疑问:为什么 Activity 页面栈方式还需要在 onActivityResumed 中设置当前 Activity 页面?
** 答疑:
当关闭 B 页面返回 A 页面时,首先 A 页面的 onResume 会先执行,然后才会调用 B 页面的 onDestroy

再来看看 ActivityStackManager 是怎么实现获取当前 Activity 页面

// ActivityStackManager.java

private WeakReference<Activity> sCurrentActivityWeakRef;

public Activity getCurrentActivity() {
    Activity currentActivity = null;
    if (sCurrentActivityWeakRef != null) {
        currentActivity = sCurrentActivityWeakRef.get();
    }
    return currentActivity;
}

public void setCurrentActivity(Activity activity) {
    sCurrentActivityWeakRef = new WeakReference<Activity>(activity);
}

public Activity getTopActivity(){
    if (activities != null && activities.size() > 0) {
        return activities.peek();
    }
    return null;
}

public void setTopActivity(Activity activity){
    if (activities != null && activities.size() > 0) {
        if (activities.search(activity) == -1) {
            activities.push(activity);
            return;
        }

        int location = activities.search(activity);
        if (location != 1) {
            activities.remove(activity);
            activities.push(activity);
        }
    }
}

4、判断应用前后台

判断应用是否在后台运行,针对前后台运行会做一些处理,比如提示用户应用运行在后台、以及应用前后台切换回调通知等。业务场景非常多,对于开发而言就是提供稳定可靠的检测前后台方法,避免出现机型不兼容问题。

AndroidProcess 提供 6 种方法判断 App 位于前台或者后台。利用
ActivityLifecycleCallbacks 实现的方式简单有效兼容性好,也不要需要申请权限。

// AppLifecycleCallback.java

private int appCount = 0;
private boolean isForground = true;

@Override
public void onActivityStarted(Activity activity) {
    appCount++;
    if (!isForground) {
        isForground = true;
        Log.e("AppLifecycle", "app into forground ");
    }
}

@Override
public void onActivityStopped(Activity activity) {
    appCount--;
    if (!isForgroundAppValue()) {
        isForground = false;
        Log.e("AppLifecycle", "app into background ");
    }
}

private boolean isForgroundAppValue() {
    return appCount > 0;
}

5、保存恢复状态值

Activity 异常退出经常需要保存恢复一些数据,ActivityLifecycleCallbacks 实现数据保存恢复也是比较简单的。

// AppLifecycleCallback.java

@Override
public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
    if (savedInstanceState != null && savedInstanceState.getBoolean("saveStateKey", false)) {
        Log.e(TAG, "localTime --> " + savedInstanceState.getLong("localTime"));
    }
}

@Override
public void onActivitySaveInstanceState(Activity activity, Bundle outState) {
    outState.putBoolean("saveStateKey", true);
    outState.putLong("localTime", System.currentTimeMillis());
}

Demo源码

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269

推荐阅读更多精彩内容