从源码看 Jetpack(2)- Lifecycle 衍生

Google Jetpack 自从推出以后,极大地改变了 Android 开发者们的开发模式,并降低了开发难度。这也要求我们对当中一些子组件的实现原理具有一定的了解,所以我就打算来写一系列 Jetpack 源码解析的文章,希望对你有所帮助

上篇文章详细讲述了 Lifecycle 的整个事件分发逻辑,本篇文章再来介绍下 Lifecycle 中几个比较容易忽略的衍生产物,基于以下版本来进行讲解

implementation "androidx.lifecycle:lifecycle-service:2.2.0"
implementation "androidx.lifecycle:lifecycle-process:2.2.0"

一、LifecycleService

LifecycleOwner 接口用于标记其实现类具备 Lifecycle 对象,即具备生命周期。而四大组件之一的 Service 本身从 启动/绑定 再到 停止,具有着类似 Activity / Fragment 从前台到退出页面之间的一系列行为,所以 Jetpack 也提供了 LifecycleService 这个 Service 的子类,用于监听 Service 的生命周期活动

LifecycleService 的源码较为简单,仅仅是在各个生命周期函数中将当前的 Event 事件转发给 ServiceLifecycleDispatcher 进行处理,由其来进行具体的事件分发。当中,onBindonStart所触发的均是 Lifecycle.Event.ON_START 事件,这是为了兼顾 startServicebindService 两种不同的情况

public class LifecycleService extends Service implements LifecycleOwner {

    private final ServiceLifecycleDispatcher mDispatcher = new ServiceLifecycleDispatcher(this);

    @CallSuper
    @Override
    public void onCreate() {
        mDispatcher.onServicePreSuperOnCreate();
        super.onCreate();
    }

    @CallSuper
    @Nullable
    @Override
    public IBinder onBind(@NonNull Intent intent) {
        mDispatcher.onServicePreSuperOnBind();
        return null;
    }

    @SuppressWarnings("deprecation")
    @CallSuper
    @Override
    public void onStart(@Nullable Intent intent, int startId) {
        mDispatcher.onServicePreSuperOnStart();
        super.onStart(intent, startId);
    }

    // this method is added only to annotate it with @CallSuper.
    // In usual service super.onStartCommand is no-op, but in LifecycleService
    // it results in mDispatcher.onServicePreSuperOnStart() call, because
    // super.onStartCommand calls onStart().
    @CallSuper
    @Override
    public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
        return super.onStartCommand(intent, flags, startId);
    }

    @CallSuper
    @Override
    public void onDestroy() {
        mDispatcher.onServicePreSuperOnDestroy();
        super.onDestroy();
    }

    @Override
    @NonNull
    public Lifecycle getLifecycle() {
        return mDispatcher.getLifecycle();
    }
}

ServiceLifecycleDispatcher 的逻辑也较为简单,内部也使用到了 LifecycleRegistry 作为 LifecycleService 的getLifecycle() 方法的返回值,这一点和 androidx.appcompat.app.AppCompatActivityandroidx.fragment.app.Fragment 保持一致

ServiceLifecycleDispatcher 将每一次的 Event 事件都包装为 DispatchRunnable 对象,然后转交由 mHandler 来执行。此外,为了保证 Lifecycle.Event 能被及时触发并保证有序性,postDispatchRunnable()方法会主动调用mLastDispatchRunnable对象的 run() 方法(如果不为 null 的话)。因为交由 Handler 执行的 Runnable 并不是可以保证就是实时完成的,为了保证 Event 值的有序性就会在有新 Event 到来时主动调用 run() 方法

/**
 * Helper class to dispatch lifecycle events for a service. Use it only if it is impossible
 * to use {@link LifecycleService}.
 */
@SuppressWarnings("WeakerAccess")
public class ServiceLifecycleDispatcher {
    private final LifecycleRegistry mRegistry;
    private final Handler mHandler;
    private DispatchRunnable mLastDispatchRunnable;

    /**
     * @param provider {@link LifecycleOwner} for a service, usually it is a service itself
     */
    public ServiceLifecycleDispatcher(@NonNull LifecycleOwner provider) {
        mRegistry = new LifecycleRegistry(provider);
        mHandler = new Handler();
    }

    private void postDispatchRunnable(Lifecycle.Event event) {
        //为了保证 Lifecycle.Event 能被及时触发并保证有序性
        //当有新的 Event 到来时,如果 mLastDispatchRunnable 不为 null
        //则主动执行 run 方法及时触发上一次的 Event,mLastDispatchRunnable 内部也做了避免重复调用的判断
        if (mLastDispatchRunnable != null) {
            mLastDispatchRunnable.run();
        }
        mLastDispatchRunnable = new DispatchRunnable(mRegistry, event);
        //将 mLastDispatchRunnable 插入到 mHandler 的队列头部
        mHandler.postAtFrontOfQueue(mLastDispatchRunnable);
    }

    /**
     * Must be a first call in {@link Service#onCreate()} method, even before super.onCreate call.
     */
    public void onServicePreSuperOnCreate() {
        postDispatchRunnable(Lifecycle.Event.ON_CREATE);
    }

    /**
     * Must be a first call in {@link Service#onBind(Intent)} method, even before super.onBind
     * call.
     */
    public void onServicePreSuperOnBind() {
        postDispatchRunnable(Lifecycle.Event.ON_START);
    }

    /**
     * Must be a first call in {@link Service#onStart(Intent, int)} or
     * {@link Service#onStartCommand(Intent, int, int)} methods, even before
     * a corresponding super call.
     */
    public void onServicePreSuperOnStart() {
        postDispatchRunnable(Lifecycle.Event.ON_START);
    }

    /**
     * Must be a first call in {@link Service#onDestroy()} method, even before super.OnDestroy
     * call.
     */
    public void onServicePreSuperOnDestroy() {
        postDispatchRunnable(Lifecycle.Event.ON_STOP);
        postDispatchRunnable(Lifecycle.Event.ON_DESTROY);
    }

    /**
     * @return {@link Lifecycle} for the given {@link LifecycleOwner}
     */
    @NonNull
    public Lifecycle getLifecycle() {
        return mRegistry;
    }

    static class DispatchRunnable implements Runnable {
        private final LifecycleRegistry mRegistry;
        final Lifecycle.Event mEvent;
        private boolean mWasExecuted = false;

        DispatchRunnable(@NonNull LifecycleRegistry registry, Lifecycle.Event event) {
            mRegistry = registry;
            mEvent = event;
        }
        
        @Override
        public void run() {
            //mWasExecuted 用于标记 run() 方法是否已经被执行过了
            //由于 Handler 和 postDispatchRunnable() 会先后调用 run() 方法
            //所以需要 mWasExecuted 来避免重复调用
            if (!mWasExecuted) {
                mRegistry.handleLifecycleEvent(mEvent);
                mWasExecuted = true;
            }
        }
    }
}

令我比较疑惑的一点就是,ServiceLifecycleDispatcher 的设计者为什么要将 Handler 作为默认的执行方式,而非通过直接调用 mRegistry.handleLifecycleEvent(mEvent)来完成事件的分发,使得如今 ServiceLifecycleDispatcher 内部的逻辑是两种调用方式都有可能被使用到,最后是通过什么方式来执行的具有不确定性

我猜测应该是为了保证 LifecycleService 的各个生命周期回调函数能够尽快完成,系统对 Service 的生命周期函数的回调操作具有超时机制,为了避免由于外部 Observer 存在耗时操作而被阻塞住,所以就不直接回调 handleLifecycleEvent 方法。但是如果 Observer 存在耗时操作的话,转交给 Handler 也一样会导致 ANR,搞不太懂

二、ProcessLifecycleOwner

ProcessLifecycleOwner 是 androidx.lifecycle:lifecycle-process:xxx 库下的一个 LifecycleOwner 实现类,可用于监听整个应用的前后台变化,在一些场景下(比如消息推送时的跳转、数据埋点)是比较有用的

使用方式如下所示:

ProcessLifecycleOwner.get().lifecycle.addObserver(object : DefaultLifecycleObserver {

    override fun onCreate(owner: LifecycleOwner) {
        Log.e("TAG", "应用被启动")
    }

    override fun onResume(owner: LifecycleOwner) {
        Log.e("TAG", "应用进入前台")
    }

    override fun onStop(owner: LifecycleOwner) {
        Log.e("TAG", "应用进入后台")
    }

})

ProcessLifecycleOwner 使用到了单例模式,获取到其唯一实例后向直接其添加 Observer 即可。需要注意的是,ProcessLifecycleOwner 是依靠于应用内所有 Activity 的生命周期的变化来定义整个应用的生命周期事件的(其实对于 Activity 的具体类型有所限制,最后会讲到),所以对于那些完全无 UI 界面的应用来说使用 ProcessLifecycleOwner 是没有意义的

再来具体看下 ProcessLifecycleOwner 是如何实现的

EmptyActivityLifecycleCallbacks

Application 包含一个很有用的方法:registerActivityLifecycleCallbacks(ActivityLifecycleCallbacks),该方法用于向外部通知应用内所有 Activity 的生命周期回调事件通知,例如,在 Activity 的 onCreate() 函数被调用前(onActivityPreCreated)、被调用时(onActivityCreated)、被调用后(onActivityPostCreated)都提供了相应的监听回调

ProcessLifecycleOwner 就使用到了registerActivityLifecycleCallbacks 方法,ProcessLifecycleOwner 的 EmptyActivityLifecycleCallbacks 实现了 ActivityLifecycleCallbacks 接口的所有方法

class EmptyActivityLifecycleCallbacks implements Application.ActivityLifecycleCallbacks {
    @Override
    public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
    }

    @Override
    public void onActivityStarted(Activity activity) {
    }

    @Override
    public void onActivityResumed(Activity activity) {
    }

    @Override
    public void onActivityPaused(Activity activity) {
    }

    @Override
    public void onActivityStopped(Activity activity) {
    }

    @Override
    public void onActivitySaveInstanceState(Activity activity, Bundle outState) {
    }

    @Override
    public void onActivityDestroyed(Activity activity) {
    }
}

使用 registerActivityLifecycleCallbacks来监听 Activity 的最大优势就是它除了可以作用于我们自定义的所有 Activity 外,还包括依赖库中的所有第三方 Activity,大多数情况下我们是无法也不会去直接修改第三方依赖库中的代码,通过系统提供的方法我们才可以比较简单地向其注入一些自定义逻辑

需要注意的是,onActivityPreCreatedonActivityPostCreated 这两类函数都是 SDK 29 时新增的默认函数,在 SDK 29 之前只包含 onActivityCreated 这类函数,为了保证兼容性,最终用到的也只有 onActivityCreated 这类函数

LifecycleDispatcher

LifecycleDispatcher 的主要逻辑是用于向应用内所有 Activity 注入一个 ReportFragment,通过 ReportFragment 来辅助获取 Activity 的生命周期事件。由外部通过调用 init(Context) 方法来进行初始化

class LifecycleDispatcher {

    private static AtomicBoolean sInitialized = new AtomicBoolean(false);

    static void init(Context context) {
        //避免重复初始化
        if (sInitialized.getAndSet(true)) {
            return;
        }
        ((Application) context.getApplicationContext())
                .registerActivityLifecycleCallbacks(new DispatcherActivityCallback());
    }

    @SuppressWarnings("WeakerAccess")
    @VisibleForTesting
    static class DispatcherActivityCallback extends EmptyActivityLifecycleCallbacks {

        @Override
        public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
            //当 Activity 被创建时,向其注入一个 ReportFragment(如果需要的话)
            ReportFragment.injectIfNeededIn(activity);
        }

        @Override
        public void onActivityStopped(Activity activity) {
        }

        @Override
        public void onActivitySaveInstanceState(Activity activity, Bundle outState) {
        }
    }

    private LifecycleDispatcher() {
    }
}

ProcessLifecycleOwner

ProcessLifecycleOwner 实现了 LifecycleOwner 接口,也用到了 LifecycleRegistry 作为其生命周期实现。其构造函数是私有的,通过静态常量来实现单例模式

public class ProcessLifecycleOwner implements LifecycleOwner {

    private final LifecycleRegistry mRegistry = new LifecycleRegistry(this);
    
    private static final ProcessLifecycleOwner sInstance = new ProcessLifecycleOwner();

    /**
     * The LifecycleOwner for the whole application process. Note that if your application
     * has multiple processes, this provider does not know about other processes.
     *
     * @return {@link LifecycleOwner} for the whole application.
     */
    @NonNull
    public static LifecycleOwner get() {
        return sInstance;
    }
    
    @NonNull
    @Override
    public Lifecycle getLifecycle() {
        return mRegistry;
    }
    
    ···

}

开放了 init(Context)函数来由外部进行变量初始化,该方法的主要逻辑是这样的:

  • 通过外部传入的 context 对象获取到 Application 对象
  • 向 Application 注册 registerActivityLifecycleCallbacks,通过 ReportFragment 来实现对所有系统版本下的 Activity 进行全局的生命周期事件监听

至此,就实现了对应用内所有 Activity 的生命周期事件监听,再之后只要再来计算处于前台的 Activity 数量的变化,就可以判断出应用所处的状态了

private Handler mHandler;

private final LifecycleRegistry mRegistry = new LifecycleRegistry(this);

ActivityInitializationListener mInitializationListener = new ActivityInitializationListener() {
            @Override
            public void onCreate() {
            }

            @Override
            public void onStart() {
                activityStarted();
            }

            @Override
            public void onResume() {
                activityResumed();
            }
        };

static void init(Context context) {
    sInstance.attach(context);
}

void attach(Context context) {
    mHandler = new Handler();
    //因为 ProcessLifecycleOwner 是针对于对整个应用的生命周期的监听
    //会执行到这一步的话说明应用肯定被启动了,此时就到了 Lifecycle.Event.ON_CREATE
    //且由于 attach 方法只会被调用一次,所以外部也只会收到一次 Lifecycle.Event.ON_CREATE 事件
    mRegistry.handleLifecycleEvent(Lifecycle.Event.ON_CREATE);
    Application app = (Application) context.getApplicationContext();
    app.registerActivityLifecycleCallbacks(new EmptyActivityLifecycleCallbacks() {

        //此方法是 SDK 29 时新增的,所以当 SDK 版本小于 29 时此方法是无效的
        @Override
        public void onActivityPreCreated(@NonNull Activity activity,
                                         @Nullable Bundle savedInstanceState) {
            //当 SDK 版本大于等于 29 时 activityStarted 和 activityResumed 这两个事件依靠于
            //EmptyActivityLifecycleCallbacks 的回调
            //当 SDK 版本小于 29 时,则需要依赖于 ReportFragment 的回调
            activity.registerActivityLifecycleCallbacks(new EmptyActivityLifecycleCallbacks() {
                @Override
                public void onActivityPostStarted(@NonNull Activity activity) {
                    activityStarted();
                }

                @Override
                public void onActivityPostResumed(@NonNull Activity activity) {
                    activityResumed();
                }
            });
        }

        @Override
        public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
            if (Build.VERSION.SDK_INT < 29) {
                //在 LifecycleDispatcher 中已经为每个 Activity 注入了 ReportFragment
                //所以此处都可以成功获取到 ReportFragment 对象并设置回调事件
                ReportFragment.get(activity).setProcessListener(mInitializationListener);
            }
        }

        @Override
        public void onActivityPaused(Activity activity) {
            activityPaused();
        }

        @Override
        public void onActivityStopped(Activity activity) {
            activityStopped();
        }
    });
}

一般情况下,一个应用的 Activity 启动流程可以概括为以下几种:

  1. 应用第一次打开时启动了 Activity A,此时会先后回调: A-onCreate、A-onStart、A-onResume
  2. 用户按 Home 键退出 Activity A,此时会先后回调:A-onPause、A-onStop
  3. 用户从后台启动了 Activity A,此时会先后回调:A-onStart、A-onResume
  4. 用户旋转了屏幕导致 Activity A 被重建,此时会先后回调:A-onPause、A-onStop、A-onDestroy、A-onCreate、A-onStart、A-onResume
  5. 用户从 Activity A 启动了 Activity B,此时会先后回调:A-onPause、B-onCreate、B-onStart、B-onResume、A-onStop
  6. 用户按返回键从 Activity B 回到 Activity A,此时会先后回调:B-onPause、A-onStart、A-onResume、B-onStop、B-onDestroy
  7. 用户按返回键退出 Activity A,此时会先后回调:A-onPause、A-onStop、A-onDestroy

可以看到,Activity 前后台切换的情况是比较复杂多样化的,且还存在 Activity 被重建的情况,所以 ProcessLifecycleOwner 就需要考虑各种情况并且将其映射为 Lifecycle.Event 的各个状态

ProcessLifecycleOwner 内部有几个变量作为状态标记位而存在

// ground truth counters
//当有 Activity 走到 Started 状态时,则 mStartedCounter 加一
//当有 Activity 走到 Stopped 状态时,则 mStartedCounter 减一
private int mStartedCounter = 0;

//当有 Activity 走到 Resumed 状态时,则 mResumedCounter 加一
//当有 Activity 走到 Paused 状态时,则 mResumedCounter 减一
private int mResumedCounter = 0;

//当发布了 ON_RESUME 事件时,值变为 false
//当发布了 ON_PAUSE 事件时,值变为 true
private boolean mPauseSent = true;

//当发布了 ON_START 事件时,值变为 false
//当发布了 ON_STOP 事件时,值变为 true
private boolean mStopSent = true;

当有 Activity 走到 onStart 状态时会调用 activityStarted()函数,而需要向外发布 ON_START 事件只在以下两种场景发生:

  1. 应用从运行开始第一次启动了 Activity。此时 mStartedCounter 从 0 递增为 1,且 mStopSent 还保持着默认值 true
  2. 应用从后台切换到了前台。此时需确保上一次发布的是 ON_STOP 事件,即 mStopSent 为 true 时,等式才能成立。因为存在这么一种特殊情况:应用只包含一个 Activity,且用户旋转了屏幕导致了该 Activity 被重建,此时 Activity 会重新走一遍生命周期流程,但对于开发者来说,Activity 还是处于前台,此时就不应该再次发布 ON_START 事件,所以 ProcessLifecycleOwner 内部对这种情况做了延时判断处理,只有上一次发布的是 ON_STOP 事件时,才会向外发布 ON_START 事件
void activityStarted() {
    mStartedCounter++;
    if (mStartedCounter == 1 && mStopSent) {
        mRegistry.handleLifecycleEvent(Lifecycle.Event.ON_START);
        mStopSent = false;
    }
}

当有 Activity 走到 onResumed 状态时会调用 activityResumed()函数,而需要向外发布 ON_RESUME 事件只在以下两种场景发生:

  1. 应用从运行开始第一次启动了 Activity。此时 mResumedCounter 从 0 递增为 1,且 mPauseSent 还保持着默认值 true
  2. 当前处于前台的 Activity 数量为 1,且上一次发布的是 ON_PAUSE 事件时(即 mPauseSent 为 true),才会发布 ON_RESUME 事件。此时一样是为了兼容用户旋转了屏幕导致了 Activity 被重建的情况
//当有 Activity 走到 onResumed 时被调用
void activityResumed() {
    mResumedCounter++;
    if (mResumedCounter == 1) {
        if (mPauseSent) {
            mRegistry.handleLifecycleEvent(Lifecycle.Event.ON_RESUME);
            mPauseSent = false;
        } else {
            mHandler.removeCallbacks(mDelayedPauseRunnable);
        }
    }
}

当有 Activity 走到 onPaused 状态时会调用 activityPaused()函数,而 mResumedCounter == 0 这个条件成立的可能原因有两种:

  1. 应用从前台退到了后台。此时需要发布 ON_PAUSE 事件
  2. 应用还保持在前台,但由于用户旋转了屏幕导致 Activity 处于重建中。所以为了避免由于第一种情况导致误判,此处会通过 Handler 来发送一个延迟消息,在 700 毫秒后(等待 Activity 重建完成)再来进行检查是否真的需要发布 ON_PAUSE 事件

因此对于开发者来说,ON_PAUSE 事件是会有一定延时的

@VisibleForTesting
static final long TIMEOUT_MS = 700; //mls

private Runnable mDelayedPauseRunnable = new Runnable() {
    @Override
    public void run() {
        dispatchPauseIfNeeded();
        dispatchStopIfNeeded();
    }
};

void activityPaused() {
    mResumedCounter--;
    if (mResumedCounter == 0) {
        mHandler.postDelayed(mDelayedPauseRunnable, TIMEOUT_MS);
    }
}

//判断是否有需要向外传递 ON_PAUSE 事件
void dispatchPauseIfNeeded() {
    if (mResumedCounter == 0) {
        //如果当前处于前台的 Activity 数量为 0,则向外传递 ON_PAUSE 事件
        mPauseSent = true;
        mRegistry.handleLifecycleEvent(Lifecycle.Event.ON_PAUSE);
    }
}

//判断是否有需要向外传递 ON_STOP 事件
void dispatchStopIfNeeded() {
    if (mStartedCounter == 0 && mPauseSent) {
        //如果当前处于 Started 状态的 Activity 数量还是为 0,则向外发送了 ON_PAUSE 事件
        //则向外传递 ON_STOP 事件
        mRegistry.handleLifecycleEvent(Lifecycle.Event.ON_STOP);
        mStopSent = true;
    }
}

当有 Activity 走到 onStopped 状态时会调用 activityStopped()函数,而需要向外发布 ON_STOP 事件只在以下一种场景发生:

  1. 应用从前台退到了后台。此时需要发布 ON_STOP 事件
void activityStopped() {
    mStartedCounter--;
    dispatchStopIfNeeded();
}

void dispatchStopIfNeeded() {
    if (mStartedCounter == 0 && mPauseSent) {
        mRegistry.handleLifecycleEvent(Lifecycle.Event.ON_STOP);
        mStopSent = true;
    }
}

ProcessLifecycleOwnerInitializer

上文还有个小细节,就是 LifecycleDispatcher 和 ProcessLifecycleOwner 两个类都需要外部传入 Context 对象以便进行初始化,但开发者在使用时其实是不需要手动初始化的,因为这个初始化过程都被隐藏在了 ProcessLifecycleOwnerInitializer 这个 ContentProvider 内部了,Application 在启动的过程中就会自动调用其 onCreate()方法来完成初始化

public class ProcessLifecycleOwnerInitializer extends ContentProvider {
    @Override
    public boolean onCreate() {
        LifecycleDispatcher.init(getContext());
        ProcessLifecycleOwner.init(getContext());
        return true;
    }

    ···
        
}

androidx.lifecycle:lifecycle-process:xxx 库中的 AndroidManifest 文件也包含了对 ProcessLifecycleOwnerInitializer 的声明,在编译时会自动合并到主项目工程中



    

    
        
    


总结

  • ProcessLifecycleOwner 一般是用于判断应用的前后台变化,依靠应用内所有 Activity 的生命周期变化来进行判断,所以对于那些完全无 UI 界面的应用来说使用 ProcessLifecycleOwner 是没有意义的
  • 由于 ReportFragment 本身的特性限制,ProcessLifecycleOwner 监听的只能是实现了 LifecycleOwner 接口的 Activity(例如 AppCompatActivity),直接继承于 android.app.Activity 的不参与判断
  • ON_CREATE 事件只会在 ProcessLifecycleOwner 初始化的时候被触发
  • 当应用刚被打开,或者是从后台切换到前台时,会依次触发 ON_START、ON_RESUME 事件
  • 当应用从前台退到后台时,会依次触发 ON_PAUSE、ON_STOP 事件
  • 由于存在用户旋转了屏幕导致 Activity 被重建的情况,ProcessLifecycleOwner 内部通过延时判断来区分这种情况,因此 ON_PAUSE、ON_STOP 这两个事件会有 700 毫秒的延迟
  • ON_DESTROY 事件永远不会被触发
  • ProcessLifecycleOwner 的初始化由 ProcessLifecycleOwnerInitializer 来隐式完成

你可能感兴趣的:(从源码看 Jetpack(2)- Lifecycle 衍生)