이전의 아키텍쳐 글들을 참조해서 멀티모듈, 모놀리식, MSA의 개념들을 먼저 공부하는 것을 추천드립니다.
데이터베이스와 같은 외부에 접근하는 구현은 생략했습니다.
멀티모듈 아키텍쳐는 Monolithic과 MSA를 상호 전환가능한 아키텍처의 기초가 되도록 설계를 했습니다. 이전의 설계한 아키텍쳐는 개발하는 과정에서 controller, service, repository의 의존성 증가되고 가독성이 떨어지는 등 아키텍쳐가 애매하게 설계가되고 상세한 기조가 부족했다고 생각했습니다.
그래서 Facade패턴을 도입하여 의존성을 낮추고 가독성을 높이는 작업도 추가했습니다. Facade패턴을 적용하는 과정에서 생각했던 부분과 상세 기조는 따로 기술할 계획입니다.
IntelliJ | JAVA17 | Spring Boot 2.7 | Gradle | JPA
RESTFul API | Facade Pattern
모듈 계층 설명
Server 모듈
Service 모듈
Core 모듈
build.gradle
plugins {
id 'java'
id 'org.springframework.boot' version '2.7.16'
id 'io.spring.dependency-management' version '1.0.15.RELEASE'
}
repositories {
mavenCentral()
}
ext {
serverEnv = System.getProperty('server.env', 'dev')
springBootVersion = '2.7.12'
}
allprojects {
apply plugin: 'java'
apply plugin: 'org.springframework.boot'
apply plugin: 'io.spring.dependency-management'
group = 'multi.module'
tasks.withType(JavaCompile) {
options.encoding = "UTF-8"
sourceCompatibility = JavaVersion.VERSION_17
targetCompatibility = JavaVersion.VERSION_17
}
repositories {
mavenCentral()
}
tasks.named('test') {
useJUnitPlatform()
}
}
subprojects {
dependencies {
// 공통 라이브러리 추가 부분
implementation 'org.springframework.boot:spring-boot-starter'
implementation 'org.springframework.boot:spring-boot-starter-web'
implementation 'org.springframework.boot:spring-boot-starter-data-jpa'
compileOnly 'org.projectlombok:lombok:1.18.24'
annotationProcessor 'org.projectlombok:lombok:1.18.24'
annotationProcessor 'org.springframework.boot:spring-boot-configuration-processor'
testImplementation 'org.projectlombok:lombok:1.18.20'
testImplementation 'org.springframework.boot:spring-boot-starter-test'
}
}
project(':module-serverA') {
dependencies {
implementation project(path: ':module-auth', configuration: 'default')
implementation project(path: ':module-service', configuration: 'default')
// 빌드관련 라이브러리, 설정 추가
}
}
project(':module-serverB') {
dependencies {
implementation project(path: ':module-service', configuration: 'default')
// 빌드관련 라이브러리, 설정 추가
}
}
project(':module-auth') {
dependencies {
implementation project(path: ':module-core', configuration: 'default')
implementation 'org.springframework.boot:spring-boot-starter-security:2.5.4'
}
}
project(':module-service') {
dependencies {
implementation project(path: ':module-core', configuration: 'default')
}
}
project(':module-core') {
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-data-redis'
implementation 'org.springframework.boot:spring-boot-starter-data-jpa'
implementation 'mysql:mysql-connector-java:8.0.29'
}
}
bootJar {
enabled = false
}
jar {
enabled = true
}
구현 설명
두가지의 구현 부분을 설명할 것이다.
Ex1을 회원이라 생각하고 Ex2 게시판이라고 했을때
먼저 Ex1(회원)에 대한 조회이다
// Controller
public class Ex1Controller {
private final Ex1Facade ex1Facade;
@GetMapping()
public ApiResult<Object> getList() {
return ApiUtils.success(ex1Facade.get());
}
}
// Facade
public class Ex1Facade {
private final Ex1Service ex1Service;
public Object get() {
return ex1Service.get();
}
}
public class Ex1Service {
public Object get() {
//구현 생략
return null;
}
}
Ex2(게시판)에 대한 생성부분이다.
Ex2Service에서 구현을 했다면 Repository에서 조회하여 재사용하지 못하지만
Service를 통해 조회함으로써 재사용하는것이다.
현재는 한개의 서버 내에서 사용가능하지만 백오피스를 개발할 경우 다른서버에서도 함수를 재사용할 수 있게 된다.
// Controller
public class Ex2Controller {
private final Ex2Facade ex2Facade;
@GetMapping()
public ApiResult<Object> create(@RequestBody Object request) {
return ApiUtils.success(ex2Facade.create(request));
}
}
// Facade
public class Ex2Facade {
private final Ex2Service ex2Service;
private final Ex1Service ex1Service;
public Object create(Object request) {
Object ob1 = ex1Service.get();
return ex2Service.create(request, ob1);
}
}
// Service
public class Ex2Service {
private final Ex2Repository ex2Repository;
public Object create(Object request, Object ob1) {
// Ex2 엔티티 생성
// ex2Repository.save(Ex2);
return null;
}
}
하나의 레포지토리로 관리하여 업데이트가 용이하고 코드를 통일하기 쉽게 만든다.
여러사람들과의 개발할 때도 편하지만 혼자 사용했을 때 코드관리가 더 용이하다는 느낌을 받아 개발인원의 규모와 상관없이 적용하는 것을 적극 추천한다.
코드 구조가 복잡해지고, 복잡해진 구조 만큼 코드를 분석하고 확인하는 과정에서 인터페이스를 거쳐 구현체들을 확인해야 하는 번거로움 발생한다.
관습적인 표현형식을 없애고 직관적으로 코드를 확인함으로써 개발속도 및 유지보수 시에 코드 파악에 있어서 시간단축이 되는 효과가 있다.
개발과정에서 코드패턴이 들쭉날쭉하여 가독성이 떨어지는 문제점 해결을 위해 사용한다.
중복된 코드보다 재사용 가능한 함수를 전역적으로 사용하기 위함이다.
비즈니스 로직 구현하는 부분에 있어서 단일 Service에 대한 많은 Repository의 의존성이 늘어나서 좀 더 객체지향적으로 개선해보았습니다. 현재 새로 진행중인 프로젝트에 리팩토링했고 기존 프로젝트와 비교해봤을때 가독성이 높이게 되었습니다. 해당 아키텍쳐를 참고하신다면 더 상세한 기조를 세워나가는 것을 추천드립니다.