관리 메뉴

김종권의 iOS 앱 개발 알아가기

[iOS] 1. Clean Architecture + MVVM 개념 확실하게 이해하기 (Entity, Use Case, Presenter) 본문

Architecture (swift)/MVVM (개념)

[iOS] 1. Clean Architecture + MVVM 개념 확실하게 이해하기 (Entity, Use Case, Presenter)

jake-kim 2021. 8. 17. 23:14

  • 소스 코드의 의존성은 안쪽을 향하게끔 설계
    • 안쪽으로 갈수록 잘 변하지 않는 요소들이기 때문
    • 안쪽의 원은 바깥쪽의 원을 모르는 상태
    • 바깥쪽의 원은 어떠한 것도 안쪽의 원에 영향을 주지 않는 구조
    • 아키텍처 설계의 가정: 사용자(Actor)의 요구사항은 변경이 많이 없고 내부적으로 Web이나 DB, UI가 자주 바뀐다고 가정하여 설계

Entity

  • Entity (=Enterprise Business Rules): Actor가 필요로 하는 데이터 모델을 의미
  • 특정 '도메인'에서 사용되는 struct 모델
  • ex) Actor가 필요로하는 Movie와 MoviesPage에 관한 Entity
struct Movie: Equatable, Identifiable {
    typealias Identifier = String
    enum Genre {
        case adventure
        case scienceFiction
    }
    let id: Identifier
    let title: String?
    let genre: Genre?
    let posterPath: String?
    let overview: String?
    let releaseDate: Date?
}

struct MoviesPage: Equatable {
    let page: Int
    let totalPages: Int
    let movies: [Movie]
}

Use case

  • Use case (= Application Business Rules)
  • Actor가 Entity를 원하는데, 이 값은 계산되거나 특정 로직에 의해서 얻어지므로 Actor가 원하는 Entity를 얻어내고 있는 '로직'을 의미
  • ex) Actor가 원하는 Entity인 MoviesPage를 얻기 위해서 필요한 '로직'
protocol SearchMoviesUseCase {
    func execute(requestValue: SearchMoviesUseCaseRequestValue,
                 cached: @escaping (MoviesPage) -> Void,
                 completion: @escaping (Result<MoviesPage, Error>) -> Void) -> Cancellable?
}

final class DefaultSearchMoviesUseCase: SearchMoviesUseCase {

    private let moviesRepository: MoviesRepository
    private let moviesQueriesRepository: MoviesQueriesRepository

    init(moviesRepository: MoviesRepository,
         moviesQueriesRepository: MoviesQueriesRepository) {

        self.moviesRepository = moviesRepository
        self.moviesQueriesRepository = moviesQueriesRepository
    }

    func execute(requestValue: SearchMoviesUseCaseRequestValue,
                 cached: @escaping (MoviesPage) -> Void,
                 completion: @escaping (Result<MoviesPage, Error>) -> Void) -> Cancellable? {

        return moviesRepository.fetchMoviesList(query: requestValue.query,
                                                page: requestValue.page,
                                                cached: cached,
                                                completion: { result in

            if case .success = result {
                self.moviesQueriesRepository.saveRecentQuery(query: requestValue.query) { _ in }
            }

            completion(result)
        })
    }
}

struct SearchMoviesUseCaseRequestValue {
    let query: MovieQuery
    let page: Int
}

Presenter

  • Entity 데이터를 그대로 표현 ('present') 하는데 필요한 계층
    • Coordinator
    • View
    • ViewModel
    • Behaviors: 특정 View의 event에 관해 적용되는 UI

External Interfaces

  • 안쪽의 원과 통신할 연결 코드 이외에는 별다른 코드를 작성하지 않는 계층 
  • 네트워크, DB 같은 것을 의미 
    • Network(URLSession)
    • DB(CoreData, UserDefaults)

* 참고

- https://blog.cleancoder.com/uncle-bob/2012/08/13/the-clean-architecture.html

Comments