[Spring Boot] [2] 7. AOP

윤경·2021년 8월 11일
2

Spring Boot

목록 보기
17/79
post-thumbnail

1️⃣ AOP가 필요한 상황

  • 모든 메소드의 호출 시간을 측정하고 싶다면?
  • 공통 관심 사항(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 를 접속하면 아래와 같은 결과 확인 가능 (첫 조회 때만 시간이 많이 걸림)

‼️ (문제점):

  • 회원가입, 회원조회에 시간을 측정하는 기능은 핵심 관심 사항이 아님.
  • 시간을 측정하는 로직은 공통 관심 사항
  • 시간을 측정하는 로직과 핵심 비즈니스의 로직이 섞여서 유지보수가 어려움.
  • 시간을 측정하는 로직을 별도의 공통 로직으로 만들기 매우 어려움.
  • 시간을 측정하는 로직을 변경할 때 모든 로직을 찾아가며 변경해야 함.

2️⃣ AOP 적용

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);
    }
}

📌 해결

  • 회원가입, 회원조회 등 핵심 관심사항과 시간을 측정하는 공통 관심 사항을 분리한다.
  • 시간을 측정하는 로직을 별도의 공통 로직으로 만들었다.
  • 핵심 관심 사항을 깔끔하게 유지할 수 있다.
  • 변경이 필요하면 이 로직만 변경하면 된다.
  • 원하는 적용 대상을 선택할 수 있다.

⬇️ AOP 적용 전 의존관계

⬇️ AOP 적용 후 의존관계

⬇️ AOP 적용 전 전체 그림

⬇️ AOP 적용 후 전체 그림


드디어 로드맵1 과정이 끝났다. 하지만 내가 제대로 배운건지 아닌지 아직도 모르는게 투성이다. 그래도 끝까지 해보자!!👩🏻‍💻 안되면 처음부터 다시 듣고 배우지 뭐

profile
개발 바보 이사 중

2개의 댓글

안녕하세요! 윤경님 블로그 보고 스프링 공부 시작하게 되었습니다!! ㅜㅜ 잘 정리된 글 감사합니다 🙇‍♀️
혹시 제 블로그에 출처 남기고 글 참고해서 공부한거 정리해도 괜찮을까요 ?

1개의 답글