網(wǎng)站首頁 編程語言 正文
定義
- 一個(gè)可以用于加速App啟動速度的庫;
- 提供在 App 啟動時(shí)初始化組件簡單、高效的方法,可以使用 App Startup 顯示的設(shè)置初始化順序;
- 提供了一個(gè) ContentProvider 來運(yùn)行所有依賴項(xiàng)的初始化,避免每個(gè)第三方庫單獨(dú)使用 ContentProvider 進(jìn)行初始化,從而提高了應(yīng)用的程序的啟動速度;
解決了什么問題
- 如果你在項(xiàng)目當(dāng)中引入了非常多的第三方庫,那么Application中的代碼就可能會變成這個(gè)樣子(這還只是我們實(shí)際項(xiàng)目的部分代碼):
class MyApplication : Application() {
override fun onCreate() {
super.onCreate()
CommonModule.init(this);
XCrash.init(this, new XCrash.InitParameters());
initQbSdk(this);
initRetrofit();
initDialogSetting();
initBugly();
initWeChat();
initUmeng();
initDoKit();
initNIM();
}
...
}
- 有些更加聰明的庫設(shè)計(jì)者,想到可以借助ContentProvider自動調(diào)用初始化接口,從而避免顯示的初始化:
//1. 繼承 ContentProvider,在onCreate中初始化
class MyProvider : ContentProvider() {
override fun onCreate(): Boolean {
context?.let {
//ContentProvider中也可以取得Context
LjyToastUtil.getInstance().init(it)
}
return true
}
//其他方法用不到,直接return null 或 return -1 即可
...
}
//2. ContentProvider是四大組件之一,需要在AndroidManifest.xml文件中進(jìn)行注冊
<application ...>
...
<provider
android:name=".MyProvider"
//authorities的值沒有固定要求,但要保證該值在整個(gè)手機(jī)上是唯一的,所以通常會使用${applicationId}作為前綴,以防止和其他應(yīng)用程序沖突
android:authorities="${applicationId}.myProvider"
android:exported="false" />
</application>
//3. 自定義的MyProvider在什么時(shí)候執(zhí)行呢? 調(diào)用流程如下:
Application.attachBaseContext() -> ContentProvider.onCreate() -> Application.onCreate()
//這是在冷啟動階段自動運(yùn)行初始化的,來看一下 Android 10 系統(tǒng)源碼
private void handleBindApplication(AppBindData data) {
...
if (!data.restrictedBackupMode) {
if (!ArrayUtils.isEmpty(data.providers)) {
// 創(chuàng)建ContentProvider
installContentProviders(app, data.providers);
}
}
...
try {
// 調(diào)用調(diào)用 Application 的 OnCreate 方法
mInstrumentation.callApplicationOnCreate(app);
} catch (Exception e) {
...
}
...
}
- 此方案的缺點(diǎn):ContentProvider會增加許多額外的耗時(shí), ContentProvider是Android四大組件之一,這個(gè)組件相對來說是比較重量級的, 也就是說,本來我的初始化操作可能是一個(gè)非常輕量級的操作,依賴于ContentProvider之后就變成了一個(gè)重量級的操作了;
如何解決問題
- 鑒于前兩者的缺點(diǎn),Google推出了App Startup
- App Startup是如何解決問題的呢?它可以將所有用于初始化的ContentProvider合并成一個(gè),從而使App的啟動速度變得更快。
使用方法
1. 引入AppStartup依賴
implementation "androidx.startup:startup-runtime:1.1.0-alpha01"
2. 實(shí)現(xiàn)App Startup庫的Initializer接口
定義一個(gè)用于執(zhí)行初始化的類,并實(shí)現(xiàn)App Startup庫的Initializer接口
class LjyToastInitializer : Initializer<Unit> {
//在create方法中執(zhí)行要初始化的代碼
override fun create(context: Context) {
LjyToastUtil.getInstance().init(context)
}
//dependencies方法用于配置當(dāng)前LjyToastInitializer是否還依賴于其他Initializer
//有的話在此配置,沒有就return emptyList()即可
override fun dependencies(): List<Class<out Initializer<*>>> {
return emptyList()
}
}
3. 在庫的AndroidManifest.xml中配置MyInitializer
<provider android:name="androidx.startup.InitializationProvider" android:authorities="${applicationId}.androidx-startup" android:exported="false" tools:node="merge"> <meta-data android:name="com.jinyang.jetpackdemo.LjyToastInitializer" android:value="androidx.startup" /> </provider>
- 當(dāng)App啟動的時(shí)候會自動執(zhí)行App Startup庫中內(nèi)置的ContentProvider,并在它的ContentProvider中會搜尋所有注冊的Initializer,然后逐個(gè)調(diào)用它們的create()方法來進(jìn)行初始化操作;
延遲初始化
- 如果不希望在啟動的時(shí)候自動初始化某個(gè)庫,而是想要在特定的時(shí)機(jī)手動初始化,這要怎么辦呢?
- 首先通過分析源碼,找到該庫初始化的Initializer的全路徑類名
- 在項(xiàng)目的AndroidManifest.xml當(dāng)中加入如下配置:
<provider android:name="androidx.startup.InitializationProvider" android:authorities="${applicationId}.androidx-startup" android:exported="false" tools:node="merge"> <meta-data android:name="com.jinyang.jetpackdemo.LjyToastInitializer" android:value="androidx.startup" tools:node="remove" /> </provider>
- 禁用單個(gè)庫就在meta-data中加入tools:node="remove"
- 禁用所有庫就是在provider標(biāo)簽中加入tools:node="remove"
- 然后在需要的地方去手動的初始化
AppInitializer.getInstance(this)
.initializeComponent(LjyToastInitializer::class.java)
- 延遲初始化也是非常有用的,可以減少 App 的啟動時(shí)間,提高啟動速度。
原文鏈接:https://zhuanlan.zhihu.com/p/405255928
相關(guān)推薦
- 2022-10-18 Python中尋找數(shù)據(jù)異常值的3種方法_python
- 2022-07-13 python版jpeg合成pdf兩種方法
- 2022-09-20 go語言VScode?see?'go?help?modules'?(exit?status?1)問題
- 2023-07-05 DateUtils 日期工具類
- 2022-05-06 input 限制輸入 小數(shù)點(diǎn)后兩位
- 2023-04-11 一文詳解C++11中的lambda函數(shù)_C 語言
- 2023-06-21 Android應(yīng)用啟動白屏處理方案詳解_Android
- 2022-12-12 C語言中帶頭雙向循環(huán)鏈表基本操作的實(shí)現(xiàn)詳解_C 語言
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支