Hilt
Hilt란 Google의 Dagger2를 기본으로 만든 의존성 주입 라이브러리이다. 따라서 Dagger2에 대한 Component, Scope를 기본적을 제공하며, 초기 프로젝트 환경 구축 비용을 크게 줄이는 것이 목적이 있다.
// Third Party Library라고 생각
class DataSource {
fun getData() {
Log.i("TAG", "데이터 다운로드..")
}
}
Hilt에 대해 알아보기 전에 Dagger2를 사용하여 의존성 주입을 사용한 간단한 예시를 구성해 보면 다음과 같다.
@Module
class DataModule {
@Provides
fun providesDataSource(): DataSource {
return DataSource()
}
}
@Component(modules = [DataModule::class])
interface DataComponent {
fun inject(mainActivity: MainActivity)
}
class App: Application() {
lateinit var dataComponent: DataComponent
override fun onCreate() {
super.onCreate()
dataComponent = DaggerDataComponent.builder()
.build()
}
}
class MainActivity : AppCompatActivity() {
@Inject
lateinit var dataSource: DataSource
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
(application as App).dataComponent
.inject(this)
// 사용하기
dataSource.getData()
}
}
<application
android:name=".App"
.
.
</application>
Dagger2로 적용되어있는 간단한 예시를 Hilt를 알아보면서 바꿔보도록 한다.
종속 항목 추가
먼저 hilt-android-gradle-plugin
플러그인을 프로젝트의 루트 build.gradle 파일에 추가한다.
buildscript {
...
dependencies {
...
classpath 'com.google.dagger:hilt-android-gradle-plugin:2.28-alpha'
}
}
app/build.gradle
파일에 종속 항복을 추가한다.
plugins {
id 'kotlin-kapt'
id 'dagger.hilt.android.plugin'
}
dependencies {
// Dagger - Hilt
implementation "com.google.dagger:hilt-android:2.41"
kapt "com.google.dagger:hilt-android-compiler:2.41"
}
Hilt는 자바8 기능을 사용한다. 대부분 자바8을 사용하기 때문에 이미 되어있는 부분일 수 있다.
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
version 범블비
안드로이드 스튜디오 범블비 버전에선 build.gradle
과 settings.gradle
이 변화하였다.
build.gradle
에 있던 이 부분이 settings.gradle
로 이동하였다.
repositories {
gradlePluginPortal()
google()
mavenCentral()
}
build.gradle
의 classpath 설정하던 부분이 다음과 같이 변경되었다.
plugins {
id 'com.android.application' version '7.1.1' apply false
id 'com.android.library' version '7.1.1' apply false
id 'org.jetbrains.kotlin.android' version '1.6.10' apply false
}
Hilt의 2.41버전이 나오면서 classpath는 build.gradle
에 다음과 같이 넣어주면 된다.
plugins {
id 'com.android.application' version '7.1.1' apply false
id 'com.android.library' version '7.1.1' apply false
id 'org.jetbrains.kotlin.android' version '1.6.10' apply false
id 'com.google.dagger.hilt.android' version '2.41' apply false
}
@HiltAndroidApp
class App: Application() {
lateinit var dataComponent: DataComponent
override fun onCreate() {
super.onCreate()
dataComponent = DaggerDataComponent.builder()
.build()
}
}
Hilt에서 @HiltAndroidApp 어노테이션을 사용하면 컴파일 타임시 표준 Component 생성에 필요한 클래스들을 초기화한다. 따라서 Hilt를 프로젝트에 적용하기 위해서는 필수적으로 요구되는 과정이다.
Dagger를 적용하였는 App 클래스를 Hilt로 변경하면 다음과 같이 코드가 줄어든다.
@HiltAndroidApp
class App: Application()
<application
android:name=".App"
.
.
</application>
Component hierachy
기존의 Dagger2는 개발자가 직접 필요한 Component들을 작성하고 상속 관계를 정의하여 사용하였지만, Hilt에서는 Android 화경에서 표준적으로 사용되는 Component들을 기본적으로 제공하고 있다.
또한 Hilt 내부적으로 제공하는 Component들의 전반적인 생명주기 또한 자동으로 관리를 해주기 때문에 초기 DI 적용을 위한 환경을 구축하는데 드는 비용을 줄일수 있다.
위와 같은 Hilt에서 제공하는 Component들은 아래와 같은 함수 호출시점에 생성되고, Destroy가 된다.
생성된 구성요소 | 생성 위치 | 제거 위치 |
---|---|---|
SingletonComponent | Application#onCreate() | Application#onDestroy() |
ActivityRetainedComponent | Activity#onCreate() | Activity#onDestroy() |
ActivityComponent | Activity#onCreate() | Activity#onDestroy() |
FragmentComponent | Fragment#onAttach() | Fragment#onDestroy() |
ViewComponent | View#super() | 제거된 뷰 |
ViewWithFragmentComponent | View#super() | 제거된 뷰 |
ServiceComponent | Service#onCreate() | Service#onDestroy() |
각 컴포넌트들은 생성 시점부터 제거되기까지 Member Injection
이 가능하고, 각 컴포넌트마다 아래와 같은 생명주기를 갖는다.
- SingletonComponent: Application 전체의 생명주기를 가진다.
- ActivityRetainComponent:
ApplicationComponent
의 하위 컴포넌트로써, Activity의 생명주기를 가진다.- Activity의 화면 회전과 같은 작업이 일어났을시 파괴되지 않고 유지된다.
- ActivityComponent:
ActivityRetainComponent
의 하위 컴포넌트로써, Activity의 생명주기를 가진다. - FragmentComponent:
ActivityComponent
의 하위 컴포넌트로써, Fragment가 Activity에 붙는(onAttach()) 시점에 생성이 되고, Fragment가 파괴되는 시점에 제더된다. - ViewComponent:
ActivityComponent
의 하위 컴포넌트로써, View의 생명주기를 가진다. - ViewWithFragmentComponent:
FragmentComponent
의 하위 컴포넌트로써, Fragment의 view 생명주기를 가진다. - ServiceComponent:
ApplicationComponent
의 하위 컴포넌트로써, Service의 생명주기를 가진다.
위와 같은 표준 Component/Scope들은 Hilt에서는 제공하고 있으며, 새로운 Component를 정의하고 싶다면 @DefineComponent
어노테이션을 사용하여 정의가 가능하다.
@InstallIn
Dagger2에서는 새로운 module을 생성하면, 사용자가 정의한 Component에 module 클래스를 직접 넣어주는 방법을 사용하였다.
@Component(modules = [DataModule::class])
interface DataComponent {
fun inject(mainActivity: MainActivity)
}
반면에 Hilt에서는 표준적으로 제공하는 Component들이 존재하기 때문에 @InstallIn 어노테이션을 사용하여 표준 Component에 Module을 install 할 수 있다.
Dagger에서 사용하였던 DataComponent를 삭제하고 DataModule에 @InstallIn을 추가한다.
@Module
@InstallIn(SingletonComponent::class)
class DataModule {
@Provides
fun providesDataSource(): DataSource {
return DataSource()
}
}
@AndroidEntryPoint
class MainActivity : AppCompatActivity() {
@Inject
lateinit var dataSource: DataSource
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
(application as App).dataComponent
.inject(this)
// 사용하기
dataSource.getData()
}
}
Dagger2에서는 의존성을 주입받을 대상을 전부 dependency graph에 지정을 해주었지만, Hilt에서는 객체를 주입할 대상에게 @AndroidEntryPoint 어노테이션을 추가만 해주면 된다. @AndroidEntryPoint를 추가할 수 있는 Android Component는 아래와 같다.
- Activity
- Fragment
- View
- Service
- BroadcastReceiver
@AndroidEntryPoint
class MainActivity : AppCompatActivity() {
@Inject
lateinit var dataSource: DataSource
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
// 사용하기
dataSource.getData()
}
}