那曲檬骨新材料有限公司

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

深入探討 Hilt的工作原理

谷歌開發(fā)者 ? 來源:Android開發(fā)者 ? 作者:Android ? 2021-10-19 15:11 ? 次閱讀
所涉主題

  • 多種 Hilt 注解協(xié)同工作并生成代碼的方式。
  • 當 Hilt 配合 Gradle 使用,Hilt Gradle 插件如何在幕后工作以改善整體體驗。

多種 Hilt 注解協(xié)同工作并生成代碼的方式

Hilt 使用注解處理器生成代碼。對注解的處理發(fā)生在編譯器將源文件轉(zhuǎn)換為 Java 字節(jié)碼期間。顧名思義,注解處理器作用于源文件中的注解。注解處理器通常會檢查注解,并根據(jù)注解類型來執(zhí)行不同的任務(wù),例如代碼檢查或生成新文件。 在 Hilt 中,三個最重要的注解就是:@AndroidEntryPoint@InstallIn 以及 @HiltAndroidApp
  • @AndroidEntryPoint
    https://dagger.dev/api/latest/dagger/hilt/android/AndroidEntryPoint.html

  • @InstallIn
    https://dagger.dev/api/latest/dagger/hilt/InstallIn.html

  • @HiltAndroidApp
    https://dagger.dev/api/latest/dagger/hilt/android/HiltAndroidApp.html

@AndroidEntryPoint

AndroidEntryPoint 在您的 Android 類中啟用字段注入,例如 Activity、Fragment、View 以及 Service。

  • AndroidEntryPoint
    https://dagger.dev/api/latest/dagger/hilt/android/AndroidEntryPoint.html

如下示例所示,通過向 PlayActivity 添加 AndroidEntryPoint 注解,即可輕松將 MusicPlayer 注入到我們的 Activity 中。
@AndroidEntryPointclass PlayActivity : AppCompatActivity() {
  @Inject lateinit var player: MusicPlayer
  // ...}

如果您使用 Gradle,您可能熟悉上文所述的簡化語法。但這并不是真實的語法,而是 Hilt Gradle 插件為您提供的語法糖。接下來我們將探討更多關(guān)于 Gradle 插件的內(nèi)容,在此之前,我們先來看看這個例子在沒有語法糖的情況下應(yīng)該是什么樣子的。
@AndroidEntryPoint(AppCompatActivity::class)class PlayActivity : Hilt_PlayActivity() {
  @Inject lateinit var player: MusicPlayer
  // ...}

現(xiàn)在,我們看到原始基類 AppCompatActivityAndroidEntryPoint 注解的真實入?yún)ⅰ?/span>PlayActivity 實際上繼承了生成的類 Hilt_PlayActivity,該類由 Hilt 注解處理器生成,并包含所有執(zhí)行注入操作需要的邏輯。針對上述內(nèi)容生成的基類,其代碼簡化示例如下:

@Generated("dagger.hilt.AndroidEntryPointProcessor")class Hilt_PlayActivity : AppCompatActivity {
  override fun onCreate() {    inject()    super.onCreate()  }
  private fun inject() {    EntryPoints.get(this, PlayActivity_Injector::class).inject(this as PlayActivity);  }}
  • AppCompatActivity
    https://developer.android.google.cn/reference/androidx/appcompat/app/AppCompatActivity

在示例中,生成的類繼承自 AppCompatActivity。然而,通常情況下生成的類會繼承傳入 AndroidEntryPoint 注解的類。這使得注入操作可以在任何您需要的基類中執(zhí)行。

生成類的主要目的是處理注入操作。為了避免字段在注入之前被意外訪問,有必要盡可能早地執(zhí)行注入操作。因此,對于 Activity,注入操作在 onCreate 中被執(zhí)行。

在 inject 方法中,我們首先需要一個注入器的實例——PlayActivity_Injector。在 Hilt 中,對于 Activity,注入器是一個入口點,我們可以使用 EntryPoints 工具類獲得一個注入器的實例。

您可能想到了,PlayActivity_Injector 也是由 Hilt 注解處理器生成的。格式如下:

@Generated("dagger.hilt.AndroidEntryPointProcessor")@EntryPoint@InstallIn(ActivityComponent::class)interface PlayActivity_Injector {
  fun inject(activity: PlayActivity)
}

生成的注入器是一個被裝載到 ActivityComponent 的 Hilt 入口點。它僅包含一個讓我們注入 PlayActivity 實例的方法。如果您曾在 Android 應(yīng)用中使用過 Dagger (不通過 Hilt),您可能會熟悉這些直接在組件上編寫的注入方法。

@InstallIn

InstallIn 用于表明模塊或者入口點應(yīng)該被裝載到哪個組件中。在如下示例中,我們將 MusicDataBaseModule 裝載到 SingletonComponent 中:
@Module@InstallIn(SingletonComponent::class)object MusicDatabaseModule {  // ...}

  • InstallIn
    https://dagger.dev/api/latest/dagger/hilt/InstallIn.html

  • SingletonComponent
    https://dagger.dev/api/latest/dagger/hilt/components/SingletonComponent.html

通過 InstallIn,應(yīng)用中任何傳遞依賴項內(nèi)都可以提供模塊和入口點。然而,部分情況下我們需要收集所有由InstallIn 注解提供的內(nèi)容以獲取每個組件的完整模塊和入口點。 Hilt 在特定的包下生成了元數(shù)據(jù)注解,以便更輕松地收集和發(fā)現(xiàn)這些由 InstallIn 注解所提供的內(nèi)容。生成的注解格式如下:
package hilt_metadata
@Generated("dagger.hilt.InstallInProcessor")@Metadata(my.database.MusicDatabaseModule::class)classMusicDatabaseModule_Metadata{}

通過將元數(shù)據(jù)放進特定的包下,Hilt 注解處理器可以輕松地在您應(yīng)用中所有的傳遞依賴項中找到生成的元數(shù)據(jù)。至此,我們可以使用元數(shù)據(jù)注解中所包含的信息來找到由 InstallIn 注解所提供內(nèi)容的自身引用。在本示例中指的是 MusicDatabaseModule

HiltAndroidApp

最后,HiltAndroidApp 注解可以讓您的 Android Application 類啟用注入。此處,您可以將其視為與 AndroidEntryPoint 注解完全相同。第一步,開發(fā)者僅需在 Application 類上添加 @HiltAndroidApp 注解。
@HiltAndroidAppclass MusicApp : Application {
  @Inject lateinit var store: MusicStore
}

  • HiltAndroidApp
    https://dagger.dev/api/latest/dagger/hilt/android/HiltAndroidApp.html

然而,HiltAndroidApp 還有另外一個重要的作用——生成 Dagger 組件。

當 Hilt 注解處理器遇到 @HiltAndroidApp 注解時,會在包裝類中生成一些列組件,該包裝類與 Application 類同名,前綴為 HiltComponents_。如果您之前使用過 Dagger,這些組件就是添加了 @Component@Subcomponent 注解的類,而在 Dagger 中通常需要您手動編寫。

a1b92714-30aa-11ec-82a8-dac502259ad0.png

為了生成這些組件,Hilt 在上述元數(shù)據(jù)包中查找所有被添加 @InstallIn 注解的類。添加了 @InstallIn 注解的模塊被放置在相應(yīng)組件聲明的模塊列表中。添加了 @InstallIn 注解的入口點被放置在聲明相應(yīng)組件的父類型的位置。

從這里開始,Dagger 處理器接管并根據(jù) @Component@Subcomponent 注解生成組件的具體實現(xiàn)。如果您曾使用過 Dagger (不通過 Hilt),那么大概率您已經(jīng)直接處理了這些類。但是,Hilt 對開發(fā)者隱藏了這種復雜操作。

這是一篇關(guān)于 Hilt 的文章,我們就不詳細介紹 Dagger 生成的代碼了。如果您有興趣,詳情請查閱:

  • Ron Shapiro 和 David Baker 的演講:

    https://www.youtube.com/watch?v=wCvXe2LsN5o

  • Dagger codegen 101 的備忘單:

    https://medium.com/androiddevelopers/dagger-code-generation-cheat-sheets-6b4fa2da4e7a

Hilt Gradle 插件

現(xiàn)在您已經(jīng)了解了 Hilt 中代碼生成的工作原理,接下來讓我們看看 Hilt Gradle 插件。Hilt Gradle 插件執(zhí)行很多有用的任務(wù),包括字節(jié)碼改寫和類路徑聚合
  • Hilt Gradle 插件

    https://dagger.dev/hilt/gradle-setup#hilt-gradle-plugin

字節(jié)碼改寫

顧名思義,字節(jié)碼改寫就是改寫字節(jié)碼的過程。與注解處理只能生成新代碼不同,字節(jié)碼改寫可以修改現(xiàn)有代碼。如果謹慎使用,這將是非常強大的功能。

為了說明我們?yōu)楹卧?Hilt 中使用字節(jié)碼改寫,讓我們回到 @AndroidEntryPoint

@AndroidEntryPoint(AppCompatActivity::class)class PlayActivity : Hilt_PlayActivity {
  override fun onCreate(…) {    val welcome = findViewById(R.id.welcome)  }
}
雖然繼承 Hilt_PlayActivity 基類在實踐中有效,但它可能會導致 IDE 報錯。由于生成的類在您成功編譯代碼后才存在,因此您經(jīng)常會在 IDE 中看到紅色波浪線。此外,您將無法享有諸如方法重載這種自動補全的能力,并且您將無法訪問基類中的方法。 失去這些功能不僅會降低您的編碼速度,而且這些紅色波浪線也會極大程度地分散您的注意力。 Hilt Android 插件通過在您的類上添加 AndroidEntryPoint 注解來啟動字節(jié)碼改寫。啟用 Hilt Android 插件后,您只需要在類上添加 @AndroidEntryPoint 注解,同時您可以使其繼承普通的基類。
@AndroidEntryPointclass PlayActivity : AppCompatActivity { // <-- 無需引用生成的基類
  override fun onCreate() {    val welcome = findViewById(R.id.welcome)  }}

由于此語法無需引用生成的基類,所以不會引起 IDE 報錯。在字節(jié)碼改寫期間,Hilt Gradle 插件會將您的基類替換為 Hilt_PlayActivity。由于此過程直接操作字節(jié)碼,對開發(fā)者是不可見的。

然而,字節(jié)碼改寫仍有一些缺點:

  • 該插件必須修改底層字節(jié)碼,而不是源代碼,這容易出錯。

  • 因為在改寫操作時字節(jié)碼已經(jīng)被編譯,所以問題通常出現(xiàn)在運行時而不是編譯時。

  • 改寫操作使調(diào)試變得復雜,因為當出現(xiàn)問題時,源文件可能并不代表當前正在執(zhí)行的字節(jié)碼。


由于這些原因,Hilt 嘗試盡可能減少依賴字節(jié)碼改寫。

類路徑聚合

最后,讓我們看看 Hilt Gradle 插件的另一個有用功能: 類路徑聚合。要了解什么是類路徑聚合,以及為什么需要它,讓我們看另一個示例。

在本示例中 :app 依賴一個獨立的 Gradle 模塊 :database:app:database 都提供了被 InstallIn 注解的模塊。

如您所見,Hilt 會在特定的 hilt_metadata 包下生成元數(shù)據(jù),在生成組件時,會用它們查找所有被添加 @InstallIn 注解的模塊。

不使用類路徑聚合的處理對于單層依賴關(guān)系仍然可以正常工作,現(xiàn)在讓我們看看當添加另一個 Gradle 模塊 :cache 作為 :database 的依賴項時會發(fā)生什么。

:cache 被編譯時,雖然它會生成元數(shù)據(jù),但在編譯 :app 時該元數(shù)據(jù)無法使用,因為它是一個傳遞依賴項。因此,Hilt 無法知曉 CacheModule,它會意外地從生成的組件中排除。 當然,您可以使用 api 而不是 implementation 聲明 :cache 的依賴關(guān)系,從而在技術(shù)層面解決這個問題,但不推薦這樣做。使用 api 不僅會讓增量構(gòu)建變得更糟糕,還把維護工作也變成一場噩夢。 這就是 Hilt Gradle 插件發(fā)揮作用的地方。

即使使用 implementation,Hilt Gradle 插件也可以自動從 :app 的傳遞依賴項中聚合所有的類。

此外,與直接使用 api 相比,Hilt Gradle 插件還具有許多優(yōu)點。

首先,對比在整個應(yīng)用中手動使用 api 依賴關(guān)系,類路徑聚合更不容易出錯并且不需要維護。您可以像往常一樣簡單地使用 implementation,其余的將由 Hilt Gradle 插件處理。

其次,Hilt Gradle 插件僅在應(yīng)用級別聚合類,因此與使用 api 不同,項目中庫的編譯不受影響。

最后,類路徑聚合為您的依賴項提供了更好的封裝,因為不可能在源文件中意外引用這些類,并且它們不會出現(xiàn)在代碼補全提示中。

總結(jié)

本文我們揭示了各種 Hilt 注解協(xié)同工作以生成代碼的方式。 我們還關(guān)注了 Hilt Gradle 插件,并了解它是如何在幕后使用字節(jié)碼改寫和類路徑聚合,讓 Hilt 的使用變得更安全、更輕松。
編輯:jq
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 處理器
    +關(guān)注

    關(guān)注

    68

    文章

    19409

    瀏覽量

    231189
  • API
    API
    +關(guān)注

    關(guān)注

    2

    文章

    1511

    瀏覽量

    62399
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4827

    瀏覽量

    69055

原文標題:Hilt 工作原理 | MAD Skills

文章出處:【微信號:Google_Developers,微信公眾號:谷歌開發(fā)者】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    鎖存器的工作原理與作用

    鎖存器(Latch)作為數(shù)字電路中的基本元件,扮演著存儲單個比特信息的關(guān)鍵角色。其獨特的功能和廣泛的應(yīng)用領(lǐng)域使其成為數(shù)字電路設(shè)計和實現(xiàn)的核心組成部分。本文將深入探討鎖存器的基本概念、工作原理、類型、主要作用以及應(yīng)用領(lǐng)域,以期為相關(guān)領(lǐng)域的技術(shù)人員提供全面而
    的頭像 發(fā)表于 02-03 16:53 ?110次閱讀

    無線收發(fā)器工作原理,無線收發(fā)器怎么使用

    無線收發(fā)器作為現(xiàn)代通信技術(shù)的重要組成部分,廣泛應(yīng)用于各個領(lǐng)域,包括無線通信、物聯(lián)網(wǎng)、遠程控制和無線傳感器網(wǎng)絡(luò)等。本文將深入探討無線收發(fā)器的工作原理,同時提供詳細的使用方法。
    的頭像 發(fā)表于 01-29 15:31 ?195次閱讀

    相位測量儀工作原理,相位測量儀怎么使用

    相位測量儀作為電子測量領(lǐng)域的重要工具,廣泛應(yīng)用于電力、通信、自動化控制等多個行業(yè)。它通過精確測量電壓、電流等信號之間的相位關(guān)系,為工程師和技術(shù)人員提供了關(guān)鍵的數(shù)據(jù)支持。本文旨在深入探討相位測量儀的工作原理,并詳細介紹其使用方法。
    的頭像 發(fā)表于 01-29 15:07 ?122次閱讀

    深入探討Linux系統(tǒng)中的動態(tài)鏈接庫機制

    異常或崩潰。為深入理解動態(tài)鏈接機制及其工作原理,我重溫了《程序員的自我修養(yǎng)》,并通過實踐演示與反匯編分析,了解了動態(tài)鏈接的過程。 本文將深入探討Linux系統(tǒng)中的動態(tài)鏈接庫機制,這其中包括但不限于全局符號介入(Global Sy
    的頭像 發(fā)表于 12-18 10:06 ?191次閱讀
    <b class='flag-5'>深入探討</b>Linux系統(tǒng)中的動態(tài)鏈接庫機制

    深入剖析測速傳感器:工作原理、種類及應(yīng)用領(lǐng)域

    的速度信息,為各種應(yīng)用場景提供了關(guān)鍵數(shù)據(jù)支持。本文將深入探討測速傳感器的工作原理、種類及其在各領(lǐng)域的應(yīng)用。 測速傳感器的工 作原理 測速傳感器的工作原理多種多樣,但核心目標都是將物體的速度信息轉(zhuǎn)化為可測量的電信號
    的頭像 發(fā)表于 08-15 18:20 ?1277次閱讀

    深入探討Linux的進程調(diào)度器

    ,以及其運行的順序。這篇文章將詳細探討Linux進程調(diào)度器的工作原理、主要算法、調(diào)度策略以及其在實際操作中的應(yīng)用。
    的頭像 發(fā)表于 08-13 13:36 ?994次閱讀
    <b class='flag-5'>深入探討</b>Linux的進程調(diào)度器

    旋轉(zhuǎn)式編碼器的工作原理和特點

    旋轉(zhuǎn)式編碼器,作為現(xiàn)代工業(yè)自動化系統(tǒng)中不可或缺的一部分,其工作原理和特點對于理解其應(yīng)用及優(yōu)化系統(tǒng)性能具有重要意義。本文將詳細闡述旋轉(zhuǎn)式編碼器的工作原理,并深入探討其特點,旨在為讀者提供全面、
    的頭像 發(fā)表于 05-29 16:01 ?1408次閱讀

    壓電換能器的工作原理、特點及應(yīng)用

    壓電換能器,作為一種能夠?qū)崿F(xiàn)電能與機械能之間相互轉(zhuǎn)換的器件,其獨特的工作原理和廣泛的應(yīng)用特點使其在多個領(lǐng)域發(fā)揮著重要作用。本文將詳細闡述壓電換能器的工作原理,并深入探討其應(yīng)用特點,以期為相關(guān)領(lǐng)域的研究和應(yīng)用提供參考。
    的頭像 發(fā)表于 05-24 16:17 ?2703次閱讀

    觸摸傳感器的工作原理、應(yīng)用及類型

    中的不可或缺性。本文將深入探討觸摸傳感器的工作原理、應(yīng)用以及主要類型,以期為讀者提供全面而深入的理解。
    的頭像 發(fā)表于 05-24 15:20 ?2598次閱讀

    計數(shù)器的結(jié)構(gòu)和工作原理

    計數(shù)器,作為一種基礎(chǔ)的電子設(shè)備,廣泛應(yīng)用于數(shù)字電路、計算機系統(tǒng)以及工業(yè)自動化等多個領(lǐng)域。它通過對輸入信號進行計數(shù)和狀態(tài)轉(zhuǎn)換,實現(xiàn)對事件發(fā)生次數(shù)的統(tǒng)計和顯示。本文將深入探討計數(shù)器的定義、結(jié)構(gòu)、工作原理以及其在各領(lǐng)域的應(yīng)用,旨在為讀者提供全面且
    的頭像 發(fā)表于 05-23 15:34 ?5381次閱讀

    達林頓晶體管的工作原理與主要特性

    在電子工程中,晶體管是一種廣泛應(yīng)用的半導體器件,它能夠控制電流的流動。在眾多晶體管類型中,達林頓晶體管(Darlington Transistor)以其獨特的工作原理和顯著的特性而備受矚目。本文將深入探討達林頓晶體管的工作原理
    的頭像 發(fā)表于 05-22 16:49 ?1596次閱讀

    隔離器的工作原理和作用

    在現(xiàn)代電子和電氣系統(tǒng)中,隔離器是一種至關(guān)重要的設(shè)備,其在信號傳輸、電源隔離、電磁干擾抑制等方面發(fā)揮著不可替代的作用。本文將深入探討隔離器的工作原理和作用,以期為讀者提供全面的理解和應(yīng)用指導。
    的頭像 發(fā)表于 05-22 16:45 ?4500次閱讀

    壓敏電阻的定義、作用及工作原理

    在電子電路的世界里,各種電子元器件扮演著各自獨特的角色,它們共同構(gòu)成了電子設(shè)備的復雜網(wǎng)絡(luò)。其中,壓敏電阻作為一種特殊的電子元器件,在電路中發(fā)揮著至關(guān)重要的作用。本文將深入探討壓敏電阻的定義、作用以及工作原理,以期為讀者提供全面而深入
    的頭像 發(fā)表于 05-22 15:12 ?3235次閱讀

    光伏儲能逆變器的工作原理及其特點

    光伏儲能逆變器是可再生能源領(lǐng)域中的關(guān)鍵設(shè)備,其工作原理和應(yīng)用范圍在推動綠色能源發(fā)展中起著至關(guān)重要的作用。本文旨在深入探討光伏儲能逆變器的工作原理及其在不同領(lǐng)域的應(yīng)用。
    的頭像 發(fā)表于 05-20 14:45 ?1359次閱讀

    激光打標機工作原理介紹

    激光打標機,作為現(xiàn)代標識技術(shù)的杰出代表,其工作原理的高效與精確性使得它在眾多行業(yè)中占據(jù)了舉足輕重的地位。今天,我們將深入探討激光打標機的工作原理及其背后的科技魅力。激光打標機的工作原理
    的頭像 發(fā)表于 03-08 19:45 ?1658次閱讀
    激光打標機<b class='flag-5'>工作原理</b>介紹
    大连百家乐食品| 百家乐官网平台注册送彩金| 百家乐辅助分析软件| 太阳城娱乐网sss977| 博彩百家乐官网最新优惠| 顶级赌场连环夺宝ios下载| 百家乐官网百家乐官网技巧| 凯斯线上娱乐| 澳门百家乐规则| 百家乐官网经验博彩正网| KK百家乐的玩法技巧和规则| 电玩百家乐官网游戏机路单| 大发888备用a99.com| 百家乐风云人物| 靖宇县| 太子百家乐的玩法技巧和规则| 百家乐官网机器图片| 视频棋牌游戏| 百家乐棋牌交友中心| 百家乐官网合理的投注法| 威尼斯人娱乐网站| 互博百家乐官网的玩法技巧和规则| 百家乐赌博机| 世嘉百家乐的玩法技巧和规则| 真人百家乐官网国际第一品牌| k7娱乐城官网| 真人百家乐开户优惠| 百家乐官网桌布尼布材质| 棋牌游戏平台开发| 百家乐平台出租家乐平台出租 | 百家乐官网赌场| 百家乐官网庄家胜率| 中山水果机定位器| 网络百家乐最安全| 516棋牌游戏中心 官方版| 真人百家乐澳门娱乐城| 百家乐官网赌博技巧大全| 皇家平台| 博九百家乐的玩法技巧和规则| 哪个百家乐最好| 免费下百家乐官网赌博软件|