이 글은 김영한님의 '스프링 입문 - 코드로 배우는 스프링 부트, 웹 MVC, DB 접근 기술'을 보고 정리한 글입니다.
https://www.inflearn.com/course/스프링-입문-스프링부트/dashboard
[무료] 스프링 입문 - 코드로 배우는 스프링 부트, 웹 MVC, DB 접근 기술 - 인프런 | 강의
스프링 입문자가 예제를 만들어가면서 스프링 웹 애플리케이션 개발 전반을 빠르게 학습할 수 있습니다., - 강의 소개 | 인프런
www.inflearn.com
1. 애플리케이션 계층 구조
일반적인 웹 애플리케이션 계층 구조
- 컨트롤러 : 웹 MVC의 컨트롤러 역할
- 서비스 : 핵심 비즈니스 로직 구현
- 리포지토리 : 데이터베이스에 접근, 도메인 객체를 DB에 저장하고 관리
- 도메인 : 비즈니스 도메인 객체. 예) 회원, 주문, 쿠폰 등 주로 데이터베이스에 저장하고 관리됨
2. 스프링 빈과 의존관계
2-1. 컴포넌트 스캔과 자동 의존관계 설정
회원 컨트롤러에 의존 관계 추가
@Controller
public class MemberController {
private final MemberService memberService;
@Autowired
public MemberController(MemberService memberService) {
this.memberService = memberService;
}
}
- 생성자에 @Autowired 를 넣어줌으로서 스프링이 연관된 객체를 스프링 컨테이너에서 찾아서 넣어줄 수 있게 한다.
- 이렇게 객체 의존관계를 외부에서 넣어주는 것을 DI(Dependency Injection) 이라고 한다.
- DI와 관련된 추가 내용은 https://wwan13.tistory.com/9 위 블로그 글을 참고하면 된다.
스프링 빈을 등록하는 두 가지 방법
- 컴포넌트 스캔과 자동 의존관계 설정
- 자바 코드로 직접 스프링 빈 등록하기
컴포넌트 스캔원리
- @Component 애노테이션이 있으면 스프링 빈으로 자동으로 등록된다.
- @Controller 컨트롤러가 스프링 빈으로 자동 등록된 이유도 컴포넌트 스캔 때문이다.
- @Controller, @Service, @Repository 와 같이 @Component를 포함하는 애노테이션도 스프링 빈으로 자동 등록된다.
생성자에 @Autowired 를 사용하면 객체 생성 시점에 스프링 컨테이너에서 해당 스프링 빈을 찾아서 주입한다. 생성자가 1개만 있으면 @Autowired 는 생략할 수 있다.
스프링은 스프링 컨테이너에 스프링 빈을 등록할 때, 기본으로 싱글톤으로 등록한다(유일하게 하나만 등록해서 공유한다) 따라서 같은 스프링 빈이면 모두 같은 인스턴스다. 설정으로 싱글톤이 아니게 설정할 수 있지만, 특별한 경우를 제외하면 대부분 싱글톤을 사용한다.
자바 코드로 직접 스프링 빈 등록하기
@Configuration
public class SpringConfig {
@Bean
public MemberService memberService() {
return new MemberService(memberRepository());
}
@Bean
public MemberRepository memberRepository() {
return new MemoryMemberRepository();
}
}
- 위 코드처럼 자바 코드와 @Bean 애노테이션을 이용해 스프링 빈으로 등록이 가능하다.
- 여기서 @Configuration 역시 @Controller 처럼 @Component를 포함하는 애노테이션으로 설정 관련된 내용을 포함하며 스프링 빈으로 자동 등록된다.
- @Bean 애노테이션을 스프링이 찾아 해당 내용을 스프링 빈으로 등록해 준다.
3. 스프링 DB접근 기술
이 강의에서는 MemoryRepository, JdbcRepository, JpaRepository, SpringDataJpaRepository 총 4개의 레파지토리를 사용한다.
여기서 중요한 것은 스프링 DI를 통해 하나의 인터페이스에 대한 구현체가 여러개 일지라도 기존 코드를 전혀 손대지 않고, 설정만으로 구현 클래스를 변경할 수 있다는 점이다.
이를 통해 OCP(Open-Closed Principle: 확장에는 열려있고, 수정 변경헤는 닫혀있다) 즉 개방 폐쇄 원칙을 지킬 수 있게 된다.
다음과 같이 간단한 설정으로 여러개의 구현체중 필요한 구현체를 선택해 사용할 수 있다.
@Configuration
public class SpringConfig {
@Bean
public MemberRepository memberRepository() {
// return new MemoryMemberRepository();
// return new JdbcMemberRepository();
// return new JpaRepository();
return new SpringDataJpaMemberRepository(dataSource);
}
}
Jpa와 SpringDataJpa에 대한 구체적인 내용은 다음 강의 정리 글을 통해 정리하겠다.
4. AOP
4.1. AOP가 필요한 상황
- 모든 메소드의 호출 시간을 측정하고 싶다면?
- 공통 관심 사항 vs 핵심 관심 사항
- 회원가입시간, 회원조회시간을 측정하고 싶다면?
MemberService 회원 조회 시간 측정 추가
public class MemberService {
public Long join(Member member) {
long start = System.currentTimeMillis();
try {
validateDuplicateMember(member); //중복 회원 검증
memberRepository.save(member);
return member.getId();
} finally {
long finish = System.currentTimeMillis();
long timeMs = finish - start;
System.out.println("join " + timeMs + "ms");
}
}
}
- 회원가입, 회원 조회에 시간을 측정하는 기능은 핵심 관심 사항이 아니다.
- 시간을 측정하는 로직은 공통 관심 사항이다.
- 시간을 측정하는 로직과 핵심 비즈니스의 로직이 섞여서 유지보수가 어렵다.
- 시간을 측정하는 로직을 별도의 공통 로직으로 만들기 매우 어렵다.
- 시간을 측정하는 로직을 변경할 때 모든 로직을 찾아가면서 변경해야 한다.
4.2. AOP 적용
- AOP: Aspect Oriented Programming
- 공통 관심 사항(cross-cutting concern) vs 핵심 관심 사항(core concern) 분리
시간 측정 AOP 등록
@Component
@Aspect
public class TimeTraceAop {
@Around("execution(* hello.hellospring..*(..))") //hellospring프로젝트 아래 모든 클래스에 적용
public Object execute(ProceedingJoinPoint joinPoint) throws Throwable {
long start = System.currentTimeMillis();
System.out.println("START: " + joinPoint.toString());
try {
return joinPoint.proceed(); // 타겟 실행
} finally {
long finish = System.currentTimeMillis();
long timeMs = finish - start;
System.out.println("END: " + joinPoint.toString()+ " " + timeMs +
}
}
}
- 회원가입, 회원 조회등 핵심 관심사항과 시간을 측정하는 공통 관심 사항을 분리한다.
- 시간을 측정하는 로직을 별도의 공통 로직으로 만들었다.
- 핵심 관심 사항을 깔끔하게 유지할 수 있다.
- 변경이 필요하면 이 로직만 변경하면 된다.
- 원하는 적용 대상을 선택할 수 있다.
AOP 적용 후 전체 그림
wwan13 - Overview
😜. wwan13 has 29 repositories available. Follow their code on GitHub.
github.com