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