parkrams
2023. 3. 30. 19:56
728x90
AOP가 필요한 상황
- 모든 메소드의 호출 시간을 측정하고 싶다면?
- 공통 관심 사항(cross-cutting concern) vs 핵심 관심 사항(core concern)
- 회원 가입 시간, 회원 조회 시간을 측정하고 싶다면?
MemberService 회원 조회 시간 측정 추가
package hello.hellospring.service;
import hello.hellospring.domain.Member;
import hello.hellospring.repository.MemberRepository;
import hello.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;
@Transactional
public class MemberService {
private final MemberRepository memberRepository;
// DI(의존성 주입) 중 생성자를 통해 들어오는 방법
// @Autowired private MemberService memberService; // 필드 주입 방법 ( 바꿔 칠 수 없어서 배드 코드 )
// set 방식이 있지만 set 방식은 누구나 호출 할 수 있기 때문에 잘 쓰지 않는다.
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
//회원 가입
public Long join(Member member){
// 같은 이름이 있는 중복 회원 x
// 옵셔널을 바로 반환하는건 배드코드
// Optional<Member> result = memberRepository.findByname(member.getName())
// // 만약 값이 있으면
// result.ifPresent(m -> {
// throw new IllegalStateException("이미 존재 하는 회원");
// });
long start = System.currentTimeMillis();
try{
validateDuplacateMember(member); // 중복 회원 검증
memberRepository.save(member);
return member.getId();
}finally {
long finish = System.currentTimeMillis();
long timeMs = finish - start;
System.out.println("join = " + timeMs + "ms");
}
//findByName을 해 그 결과는 옵셔널 멤버니까 옵셔널 멤버 . ifPresent 해서 값을 하나씩 비교~
// 값이 존재 하면 이미 존재하는 회원 출력
}
private void validateDuplacateMember(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);
}
}
- long start = System.currentTimeMillis(); < ms 단위 시간 메서드
문제
- 회원가입, 회원 조회에 시간을 측정하는 기능은 핵심 관심 사항이 아니다.
- 시간을 측정하는 로직은 공통 관심 사항이다.
- 시간을 측정하는 로직과 핵심 비즈니스의 로직이 섞여서 유지보수가 어렵다.
- 시간을 측정하는 로직을 별도의 공통 로직으로 만들기 매우 어렵다.
- 시간을 측정하는 로직을 변경할 때 모든 로직을 찾아가면서 변경해야 한다.
AOP 적용
- AOP : Aspect Oriented Programming
- 공통 관심 사항 (cross-cutting concern) vs 핵심 관심 사항 (core concern) 분리
시간 측정 AOP 등록
- SpringConfig에 AOP를 @Bean으로 등록 하면 메서드를 보고 스프링 빈에 등록 되어 쓰이는 구나 알 수 있다.
- bean 등록 하고 Class에 @Component 해주면 에러 발생
- @Around문법
// hello.hellospring 패키지 하위 쪽 ( 자식들 ) 한테 다 적용하라는 뜻
@Around("execution(* hello.hellospring..*(..))")
package hello.hellospring.aop;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class TimeTraceAop {
// hello.hellospring 패키지 하위 쪽 ( 자식들 ) 한테 다 적용하라는 뜻
@Around("execution(* hello.hellospring..*(..))")
// 호출 될 때마다 조인 포인트에다가 중간에서 인터셉트가 걸리는 것
// return에 다른 메서드를 넣어서 다른 작업도 가능하게 가능 함
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 + "ms");
}
}
}
해결
- 회원가입, 회원 조회등 핵심 관심사항과 시간을 측정하는 공통 관심 사항을 분리한다.
- 시간을 측정하는 로직을 별도의 공통 로직으로 만들었다.
- 핵심 관심 사항을 깔끔하게 유지할 수 있다.
- 변경이 필요하면 이 로직만 변경하면 된다.
- 원하는 적용 대상을 선택할 수 있다.
스프링 AOP 동작 방식 설명
AOP적용 전 의존 관계
AOP적용 후 의존 관계
AOP적용 전 전체 그림
AOP적용 후 전체 그림
- 실제 Proxy가 주입되는지 콘솔에 출력해서 확인하기
728x90