- 모든 메소드의 호출 시간을 측정하고 싶다면?
- 공통 관심 사항(cross-cutting concern) vs 핵심 관심 사항(core concern) ( ➡️ 이렇게 분리 )
- 회원 가입 시간, 회원 조회 시간을 측정하고 싶다면?
✔️ service/MemberService.java MemberService 회원 조회 시간 측정을 추가해보자.
package com.example.hellospring.service;
import com.example.hellospring.domain.Member;
import com.example.hellospring.repository.MemberRepository;
import com.example.hellospring.repository.MemoryMemberRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.util.List;
import java.util.Optional;
//@Service
@Transactional
public class MemberService {
private final MemberRepository memberRepository;
// @Autowired
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
// 회원가입
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");
}
}
private void validateDuplicateMember(Member member) {
memberRepository.findByName(member.getName())
// 멤버가 존재하는지 확인 후 존재한다면 "이미 존재하는 회원입니다." 출력
.ifPresent(m -> {
throw new IllegalStateException("이미 존재하는 회원입니다.");
});
}
// 전체회원 조회
public List<Member> findMembers() {
long start = System.currentTimeMillis();
try {
return memberRepository.findAll();
} finally {
long finish = System.currentTimeMillis();
long timeMs = finish - start;
System.out.println("findMembers " + timeMs + "ms");
}
}
public Optional<Member> findOne(Long memberId) {
return memberRepository.findById(memberId);
}
}
⬇️ 자, 이제 초가 나오는지 확인해보자
⬇️ 회원 조회도 잘 되는지 확인해보자.
(서버를 실행시키고 http://localhost:8080/members 를 접속하면 아래와 같은 결과 확인 가능 (첫 조회 때만 시간이 많이 걸림)
‼️ (문제점):
- 회원가입, 회원조회에 시간을 측정하는 기능은 핵심 관심 사항이 아님.
- 시간을 측정하는 로직은 공통 관심 사항
- 시간을 측정하는 로직과 핵심 비즈니스의 로직이 섞여서 유지보수가 어려움.
- 시간을 측정하는 로직을 별도의 공통 로직으로 만들기 매우 어려움.
- 시간을 측정하는 로직을 변경할 때 모든 로직을 찾아가며 변경해야 함.
AOP: Aspect Oriented Programming
공통 관심 사항(cross-cutting concern) vs 핵심 관심 사항(core concern) 분리
✔️ 시간 측정 AOP를 등록하자! src/main/java/com.example.hellospring/aop 패키지 만들기. 그리고 그 안에 TimeTraceAop.java 클래스를 생성하자.
package com.example.hellospring.aop;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
@Component
@Aspect
public class TimeTraceAop {
// @Around: 타겟팅
// com... 이 패키지 안에는 다 적용하겠다는 것
@Around("execution(* com.example.hellospring..*(..))")
// joinPoint가 가진 메소드로 조작할 수 있음.
public Object execute(ProceedingJoinPoint joinPoint) throws Throwable {
long start = System.currentTimeMillis();
System.out.println("START: " + joinPoint.toString());
try {
// proceed(): 다음 메소드로 진행
return joinPoint.proceed();
} finally {
long finish = System.currentTimeMillis();
long timeMs = finish - start;
System.out.println("END: " + joinPoint.toString() + " " + timeMs + "ms");
}
}
}
⬇️ 결과를 확인해보자
(서버를 실행 시킨 뒤 localhost:8080에서 회원을 등록하고 조회해보기) ⬆️ 이렇게 START, END로 실행되어야 할 메소드가 잘 수행되고 있는 것을 볼 수 있음!
‼️ 아, MemberService.java를 원상복구 시켜놓자
package com.example.hellospring.service;
import com.example.hellospring.domain.Member;
import com.example.hellospring.repository.MemberRepository;
import com.example.hellospring.repository.MemoryMemberRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.util.List;
import java.util.Optional;
//@Service
@Transactional
public class MemberService {
private final MemberRepository memberRepository;
// @Autowired
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
// 회원가입
public Long join(Member member) {
// 중복회원 검증
validateDuplicateMember(member);
memberRepository.save(member);
return member.getId();
}
private void validateDuplicateMember(Member member) {
memberRepository.findByName(member.getName())
// 멤버가 존재하는지 확인 후 존재한다면 "이미 존재하는 회원입니다." 출력
.ifPresent(m -> {
throw new IllegalStateException("이미 존재하는 회원입니다.");
});
}
// 전체회원 조회
public List<Member> findMembers() {
return memberRepository.findAll();
}
public Optional<Member> findOne(Long memberId) {
return memberRepository.findById(memberId);
}
}
📌 해결
- 회원가입, 회원조회 등 핵심 관심사항과 시간을 측정하는 공통 관심 사항을 분리한다.
- 시간을 측정하는 로직을 별도의 공통 로직으로 만들었다.
- 핵심 관심 사항을 깔끔하게 유지할 수 있다.
- 변경이 필요하면 이 로직만 변경하면 된다.
- 원하는 적용 대상을 선택할 수 있다.
드디어 로드맵1 과정이 끝났다. 하지만 내가 제대로 배운건지 아닌지 아직도 모르는게 투성이다. 그래도 끝까지 해보자!!👩🏻💻 안되면 처음부터 다시 듣고 배우지 뭐
안녕하세요! 윤경님 블로그 보고 스프링 공부 시작하게 되었습니다!! ㅜㅜ 잘 정리된 글 감사합니다 🙇♀️
혹시 제 블로그에 출처 남기고 글 참고해서 공부한거 정리해도 괜찮을까요 ?