Files
2nd/10_Wiki/Topics/Coding/Android_Modularization.md
T
2026-05-09 21:08:02 +09:00

4.7 KiB

id, title, category, status, source_trust_level, verification_status, created_at, updated_at, tags, tech_stack, applied_in, aliases
id title category status source_trust_level verification_status created_at updated_at tags tech_stack applied_in aliases
android-modularization Android Modularization — Feature / Core / App Coding draft B conceptual 2026-05-09 2026-05-09
android
gradle
modularization
vibe-coding
language applicable_to
Kotlin / Gradle
Android
feature module
core module
build time
dynamic feature

Android Modularization

한 모듈 앱은 빌드 시간과 응집도가 폭발한다. feature : core : data : app 4계층 분리 + 단방향 의존성 = 빠른 빌드 + 명확한 책임. 잘못된 의존성은 cyclical → gradle 폭사.

📖 핵심 개념

일반적 4계층:

  • app: Application class, AppNav. 모든 feature import.
  • feature:xxx: 한 feature 의 UI + ViewModel.
  • core:: 공통 (ui-kit, designsystem, network, database).
  • data:xxx: repository, remote, local.

규칙: 모든 의존성 위에서 아래로. feature 끼리는 직접 X (app 만 안다).

💻 코드 패턴

Project 구조

:app
:feature:home
:feature:profile
:feature:order
:core:ui (Compose theme, components)
:core:network (Retrofit, OkHttp)
:core:database (Room)
:core:common
:data:user (UserRepo)
:data:order (OrderRepo)

settings.gradle.kts

include(":app")
include(":feature:home", ":feature:profile", ":feature:order")
include(":core:ui", ":core:network", ":core:database", ":core:common")
include(":data:user", ":data:order")

feature module — build.gradle.kts

plugins {
    alias(libs.plugins.android.library)
    alias(libs.plugins.kotlin.android)
    alias(libs.plugins.hilt)
    alias(libs.plugins.ksp)
}

android {
    namespace = "com.example.feature.profile"
    compileSdk = 34
    defaultConfig { minSdk = 26 }
}

dependencies {
    implementation(project(":core:ui"))
    implementation(project(":core:common"))
    implementation(project(":data:user"))
    // feature 끼리 의존 X
    implementation(libs.compose.foundation)
    implementation(libs.hilt.android)
    ksp(libs.hilt.compiler)
}

Convention plugin — 중복 제거

// build-logic/src/main/kotlin/AndroidFeatureConventionPlugin.kt
class AndroidFeatureConventionPlugin : Plugin<Project> {
    override fun apply(target: Project) = with(target) {
        pluginManager.apply("com.android.library")
        pluginManager.apply("org.jetbrains.kotlin.android")
        pluginManager.apply("com.google.dagger.hilt.android")

        extensions.configure<LibraryExtension> {
            compileSdk = 34
            defaultConfig { minSdk = 26 }
            compileOptions { ... }
        }
        dependencies {
            "implementation"(project(":core:ui"))
            "implementation"(project(":core:common"))
        }
    }
}

// feature/profile/build.gradle.kts
plugins { id("android.feature") }

app — composition root

@HiltAndroidApp
class App : Application()

// AppNav.kt
@Composable
fun AppNav(nav: NavHostController) {
    NavHost(nav, startDestination = HomeRoute) {
        homeGraph(nav)        // feature:home extension
        profileGraph(nav)     // feature:profile extension
        orderGraph(nav)       // feature:order extension
    }
}

각 feature 가 NavGraphBuilder extension 을 export.

Build cache + parallel

# gradle.properties
org.gradle.parallel=true
org.gradle.caching=true
org.gradle.configureondemand=true
android.enableJetifier=false
kotlin.incremental.useClasspathSnapshot=true

🤔 의사결정 기준

앱 크기 모듈화
<10 화면 단일 모듈 OK
10-30 feature 별 분리 시작
30+ 4계층 fully modular
팀 다수 명확 ownership boundary 로 모듈
Dynamic delivery feature module = on-demand download

안티패턴

  • feature ↔ feature 직접 의존: cyclical 가능. app 또는 navigation contract 통해서.
  • core 가 feature import: 역방향. core 는 가장 아래.
  • 거대 :common 에 모든 것: 어떤 변경도 모든 모듈 재빌드. 잘게 분리.
  • 모듈마다 다른 minSdk / compileSdk: 일관성 깨짐. convention plugin.
  • 각 모듈에 같은 dependency 중복 선언: convention plugin 으로.
  • 순환 의존 발견 늦음: gradle build 시 즉시 실패. 의존 그래프 시각화 (Gradle Dependency Insight).
  • feature module 이 직접 Application class 참조: app 의존성 역방향.

🤖 LLM 활용 힌트

  • 신규 Android = 4계층 modular 출발.
  • convention plugin 으로 boilerplate 제거.
  • feature 끼리는 NavGraphBuilder extension 으로만 통신.

🔗 관련 문서