简单来说,分为三步
-
gradle 文件引入App Startup 库。
-
自定义一个用于初始化的 Initializer。
-
将自定义 Initializer 配置到 AndroidManifest.xml 当中。
第一步,在 build.gradle 文件添加依赖
1dependencies {
2 implementation “androidx.startup:startup-runtime:1.0.0”
3}
第二步:自定义实现 Initializer 类
主要有两个方法
-
T create(@NonNull Context context) 初始化一个组件,返回给 Application
-
List<class<? extends=“” initializer>> dependencies() 当前的 Initializer 依赖于那些 Initializers,通过这个可以确定先后启动的顺序
我们以官方的例子来讲解
1// Initializes WorkManager.
2class WorkManagerInitializer : Initializer {
3 override fun create(context: Context): WorkManager {
4 val configuration = Configuration.Builder().build()
5 WorkManager.initialize(context, configuration)
6 return WorkManager.getInstance(context)
7 }
8 override fun dependencies(): List<Class<out Initializer<*>>> {
9 // No dependencies on other libraries.
10 return emptyList()
11 }
12}
WorkManagerInitializer 返回一个 WorkManager,它不需要依赖于其他的 Initializer,直接返回 emptyList() 即可。
如果需要依赖其他的 Initializer,重写 dependencies 方法,返回即可。如下面的 ExampleLoggerInitializer 依赖于 WorkManagerInitializer
1// Initializes ExampleLogger.
2class ExampleLoggerInitializer : Initializer {
3 override fun create(context: Context): ExampleLogger {
4 // WorkManager.getInstance() is non-null only after
5 // WorkManager is initialized.
6 return ExampleLogger(WorkManager.getInstance(context))
7 }
8
9 override fun dependencies(): List<Class<out Initializer<*>>> {
10 // Defines a dependency on WorkManagerInitializer so it can be
11 // initialized after WorkManager is initialized.
12 return listOf(WorkManagerInitializer::class.java)
13 }
14}
15
16class ExampleLogger(val workManager: WorkManager){
17
18}
第三步:在 AndroidManifest 里面配置自定义的 InitializationProvider
1<provider
2 android:name=“androidx.startup.InitializationProvider”
3 android:authorities=“${applicationId}.androidx-startup”
4 android:exported=“false”
5 tools:node=“merge”>
6
7 <meta-data android:name=“com.xj.anchortask.appstartup.ExampleLoggerInitializer”
8 android:value=“androidx.startup” />
9
它是有固定格式的,配置者只需要配置 meta-data 中的 name 即可。 android:name=“com.xj.anchortask.appstartup.ExampleLoggerInitializer” 这里的 name 是我们自定义的 Initializer 全路径。
程序运行跑起来,可以看到以下输出结果,符合我们的预期
2021-04-17 17:48:42.049 28059-28059/com.xj.anchortask I/AnchorTaskApplication: attachBaseContext:
2021-04-17 17:48:42.077 28059-28059/com.xj.anchortask I/AnchorTaskApplication: create: WorkManagerInitializer init
2021-04-17 17:48:42.077 28059-28059/com.xj.anchortask I/AnchorTaskApplication: create: ExampleLoggerInitializer init
2021-04-17 17:48:42.084 28059-28059/com.xj.anchortask I/AnchorTaskApplication: onCreate:
========================================================================
上面我们讲解了 AppStartUp 的基本使用步骤,如果我们不想在 Application onCreate 之前执行我们的 ExampleLoggerInitializer,要怎么使用呢?
其实很简单,
-
第一步,在 AndroidManifest InitializationProvider 中移除 移除 <meta-data 标签
-
在代码中调用 AppInitializer initializeComponent 方法初始化
1<provider
2 android:name=“androidx.startup.InitializationProvider”
3 android:authorities=“${applicationId}.androidx-startup”
4 android:exported=“false”
5 tools:node=“merge”>
6
7
1AppInitializer.getInstance(context).initializeComponent(ExampleLoggerInitializer::class.java)
App start up 源码分析
我们首先来看一下他的结构,只有简单的几个类
Initializer这个接口就没有必要说了,很简单,只有两个方法。
InitializationProvider 继承了 ContentProvider,借助了 ContentProvider 会在 Application onCreate 之前执行的特点。来执行一些初始化操作。
1public final class InitializationProvider extends ContentProvider {
2 @Override
3 public boolean onCreate() {
4 Context context = getContext();
5 if (context != null) {
6 AppInitializer.getInstance(context).discoverAndInitialize();
7 } else {
8 throw new StartupException(“Context cannot be null”);
9 }
10 return true;
11 }
12
13 ----
14
15}
我们可以看到在 onCreate 方法中调用 AppInitializer discoverAndInitialize 方法进行初始化。
-
找到 AndroidManifest InitializationProvider 下的 meta 便签
-
判断 meta 便签下 value 的值是不是 androidx.startup
-
判断是不是实现 Initializer 接口,是的话,执行 doInitialize 方法
1void discoverAndInitialize() {
2 try {
3 Trace.beginSection(SECTION_NAME);
4 ComponentName provider = new ComponentName(mContext.getPackageName(),
5 InitializationProvider.class.getName());
6 ProviderInfo providerInfo = mContext.getPackageManager()
7 .getProviderInfo(provider, GET_META_DATA);
8 Bundle metadata = providerInfo.metaData;
9 String startup = mContext.getString(R.string.androidx_startup);
10 // 找到 metadata 标签
11 if (metadata != null) {
12 Set<Class<?>> initializing = new HashSet<>();
13 Set keys = metadata.keySet();
14 for (String key : keys) {
15 String value = metadata.getString(key, null);
16 // 判断 value 的值是不是 androidx.startup
17 // 判断是不是实现了 Initializer 接口,是的话,反射初始化
18 if (startup.equals(value)) {
19 Class<?> clazz = Class.forName(key);
20 if (Initializer.class.isAssignableFrom(clazz)) {
21 Class<? extends Initializer<?>> component =
22 (Class<? extends Initializer<?>>) clazz;
23 mDiscovered.add(component);
24 if (StartupLogger.DEBUG) {
25 StartupLogger.i(String.format(“Discovered %s”, key));
26 }
27 doInitialize(component, initializing);
28 }
29 }
30 }
31 }
32 } catch (PackageManager.NameNotFoundException | ClassNotFoundException exception) {
33 throw new StartupException(exception);
34 } finally {
35 Trace.endSection();
36 }
37}
doInitialize 方法
1 T doInitialize(
2 @NonNull Class<? extends Initializer<?>> component,
3 @NonNull Set<Class<?>> initializing) {
4 synchronized (sLock) {
5 boolean isTracingEnabled = Trace.isEnabled();
6 try {
7 if (isTracingEnabled) {
8 // Use the simpleName here because section names would get too big otherwise.
9 Trace.beginSection(component.getSimpleName());
10 }
11 if (initializing.contains(component)) {
12 String message = String.format(
13 “Cannot initialize %s. Cycle detected.”, component.getName()
14 );
15 throw new IllegalStateException(message);
16 }
17 Object result;
18 if (!mInitialized.containsKey(component)) {
19 initializing.add(component);
20 try {
21 Object instance = component.getDeclaredConstructor().newInstance();
22 Initializer<?> initializer = (Initializer<?>) instance;
23 List<Class<? extends Initializer<?>>> dependencies =
24 initializer.dependencies();
25
26 if (!dependencies.isEmpty()) {
27 for (Class<? extends Initializer<?>> clazz : dependencies) {
28 if (!mInitialized.containsKey(clazz)) {
29 doInitialize(clazz, initializing);
30 }
31 }
32 }
33 if (StartupLogger.DEBUG) {
34 StartupLogger.i(String.format(“Initializing %s”, component.getName()));
35 }
36 result = initializer.create(mContext);
37 if (StartupLogger.DEBUG) {
38 StartupLogger.i(String.format(“Initialized %s”, component.getName()));
39 }
40 initializing.remove(component);
41 mInitialized.put(component, result);
42 } catch (Throwable throwable) {
43 throw new StartupException(throwable);
44 }
45 } else {
46 result = mInitialized.get(component);
47 }
48 return (T) result;
资源分享
- 最新大厂面试专题
这个题库内容是比较多的,除了一些流行的热门技术面试题,如Kotlin,数据库,Java虚拟机面试题,数组,Framework ,混合跨平台开发,等
- 对应导图的Android高级工程师进阶系统学习视频
最近热门的,NDK,热修复,MVVM,源码等一系列系统学习视频都有!
43 throw new StartupException(throwable);
44 }
45 } else {
46 result = mInitialized.get(component);
47 }
48 return (T) result;
资源分享
- 最新大厂面试专题
这个题库内容是比较多的,除了一些流行的热门技术面试题,如Kotlin,数据库,Java虚拟机面试题,数组,Framework ,混合跨平台开发,等
[外链图片转存中…(img-wSGVbD14-1718769299314)]
- 对应导图的Android高级工程师进阶系统学习视频
最近热门的,NDK,热修复,MVVM,源码等一系列系统学习视频都有!
[外链图片转存中…(img-1UWC3rqh-1718769299315)]