[Logging] 로그 출력을 효율적으로 생성 및 추적하기 7편 - Spring AOP

2025. 1. 6. 15:21·Spring/Logging
728x90
반응형
SMALL

Spring AOP

수 많은 문제점을 거쳐서 드디어 결론에 도달했다... 결국에는 Spring에서 제공하는 AOP의 동작방식과 강력한 지원 기능에 대해 설명하려고 달려왔던것 같았다 ㅎㅎ

처음 다뤘던 내용과 같이 애플리케이션의 비즈니스 로직은 핵심 기능과 부가 기능으로 분리할 수 있다.
핵심 기능은 도메인의 객체가 수행하는 고유의 기능이다.
부가 기능은 핵심 기능을 보조하기위해 제공하는 기능이다. 예를 들어서 로그 추적 로직, 트랜잭션 기능 등이 이에 해당된다.
핵심 기능은 각 객체의 고유 기능이기에 중복적인 수행이 아니지만, 부가 기능은 동일한 기능을 수행하기 때문에 각 도메인마다 중복되는 기능이 수행된다.
객체지향적인 코드를 짜기 위해서는 중복된 내용은 하나의 추상화를 통해 공통적으로 사용할 수 있는 하나의 객체로 생성할 수 있고 이를 상속 또는 합성을 통해 사용하는 방법을 수행할 수 있다.

이 부가 기능을 적용하기 위해 해결해야 하는 문제를 정리하면 다음과 같다.

  1. 부가 기능 적용을 위해 많은 수정이 필요하다.
  2. 부가 기능 적용을 위해 많은 반복 작업이 필요하다.
  3. 부가 기능을 변경하게 되면 많은 수정 및 중복 작업이 필요하다.
  4. 부가 기능을 적용하는 대상을 변경하면 많은 수정이 필요하다.

이 문제를 하나의 추상체로 만든것이 Aspect이다. Aspect에서는 부가 기과 부가 기능을 어디에 적용할 것인지 정의할 수 있다.
그리고 스프링에서는 AOP(Aspect Oriented Programming)라는 이름으로 이를 제공한다.

AOP 적용 방식 - 컴파일 타임 위빙

Java 파일을 컴파일 하는 과정에서 위빙이라는 개념을 적용하여 클래스 파일을 만드는 시점에 부가 기능 로직을 추가하게 된다.
부가 기능 로직을 추가하는 단계에서 AspectJ 컴파일러는 Aspect를 확인해서 해당 클래스가 적용할 대상인지 확인 부가 기능 로직을 적용시킨다.
이 과정에서는 특별한 컴파일러가 복잡한 과정을 거쳐야 한다는 단점이 있다.

AOP 적용 방식 - 로드타임 위빙

자바를 실행하면 .class파일을 JVM 내부의 Class Loader에 저장하게된다. 이때 클래스 로더에 적재하기 이전에 파일을 조작할 수 있다.
이 시점에 부가 기능을 적용 하는 것을 로드 타임 위빙이라고 한다.
이때 클래스 로더 조기를 지정해주는 과정이 필요한데 이 방법또한 복잡하다.

AOP 적용 방식 - 런타임 위빙

런타임 위빙은 클래스를 컴파일하거나 로딩 하는 과정 이후에 적용하는 방식이다.
이때 스프링에서 빈 객체를 생성하고 스프링 컨테이너 내부에 빈의 의존성을 연결하기 이전 과정에서 실제 객체를 프록시 객체로 바꿔치기하여 AspectJ 모듈을 적용시킬 수 있다.

AOP 적용 시점

AOP는 메서 실행 시점에서 사용할 수 있다.
프록시는 메서드를 오버 라이딩하여 사용하기 때문에 생성자나 static 메서드, 필드 값에는 프록시를 적용하여 사용할 수 없다.
런타임에 빈 객체를 사용하여 프록시 객체를 사용하기 때문에 빈으로 등록되지 않은 객체는 사용이 불가능하다.

AOP 용어

  • Join Point
    • AOP를 적용할 수 있는 메서드 포인트를 말한다
  • Point Cut
    • 조인 포인 중에서 어드바이스가 적용될 위치를 설정할 수 있다
    • AspectJ 표현식을 사용한다
  • Target
    • 어드바이스를 받는 객체, 포인트 컷으로 결정한다
  • Advice
    • 특정 Join Point에서 수행할 기능에 대해 정의한다
    • Around, Before, After등의 어노테이션을 사용하여 프록시 객체의 수행 지점을 설정할 수 있다.
  • Aspect
    • Advice + PointCut의 기능
    • Aspect 어노테이션을 적용한다.
    • 여러 Advice와 PointCut이 존재한다.
  • Advisor
    • 하나의 Advice + PointCut
    • Spring AOP 에서만 적용하는 개념이다.
  • Weaving
    • PointCut으로 결정한 타겟의 Join Point에 Advice를 적용하는 것
    • 위빙을 통해 기존 코드를 수정하지 않고 부가 기능을 추가할 수 있다.

예제1

@Around 어노테이션을 사용하여 포인트컷을 설정한다.
logProcess는 @Aspect를 적용하여 어드바이스가 된다.
execution(* com.example.loggingtest.domain.user..*(..)) 이 구문은 AspectJ의 표현식이다.
해당 표현식 내의 모든 메서드는 프록시 객체로 생성되어 AOP적용 대상이 된다.


@Slf4j
@Aspect
@Component
public class AspectV1 {
    @Around("execution(* com.example.loggingtest.domain.user..*(..))")
    public Object logProcess(ProceedingJoinPoint joinPoint) throws Throwable {
        log.info("[log] {}", joinPoint.getSignature().getName());
        return joinPoint.proceed();
    }
}

예제2

pointCut을 분리해서 Around의 매개변수로 적용할 수 있다.

@Pointcut("execution(* com.example.loggingtest..*(..))")
    public void pointCut() {}

    @Around("pointCut()")
    public Object logProcess(ProceedingJoinPoint joinPoint) throws Throwable {
        log.info("[log] {}", joinPoint.getSignature().getName());
        return joinPoint.proceed();
    }
728x90
반응형
SMALL

'Spring > Logging' 카테고리의 다른 글

[Logging] 내 프로젝트에 Logging 적용해서 로그 출력하기  (0) 2025.01.13
[Logging] 로그 출력을 효율적으로 생성 및 추적하기 6편 - Proxy  (1) 2025.01.06
[Logging] 로그 출력을 효율적으로 생성 및 추적하기 5편 - Template Method  (1) 2025.01.06
[Logging] 로그 출력을 효율적으로 생성 및 추적하기 4편 ThreadLocal사용기  (0) 2025.01.06
[Logging] 로그 출력을 효율적으로 생성 및 추적하기 3편  (1) 2024.12.26
'Spring/Logging' 카테고리의 다른 글
  • [Logging] 내 프로젝트에 Logging 적용해서 로그 출력하기
  • [Logging] 로그 출력을 효율적으로 생성 및 추적하기 6편 - Proxy
  • [Logging] 로그 출력을 효율적으로 생성 및 추적하기 5편 - Template Method
  • [Logging] 로그 출력을 효율적으로 생성 및 추적하기 4편 ThreadLocal사용기
공부하고 기억하는 공간
공부하고 기억하는 공간
IT 비전공자로 시작하여 훌륭한 개발자가 되기 위해 공부하고 있는 공간입니다. 틀린 내용이나 부족한 부분이 있으면 댓글로 알려주세요 바로 수정하겠습니다.
    250x250
  • 공부하고 기억하는 공간
    IT - railroad
    공부하고 기억하는 공간
  • 전체
    오늘
    어제
    • 분류 전체보기 (325)
      • 면접 준비 (22)
        • OS (6)
        • Spring Security (0)
        • Java (3)
        • DB (11)
        • Network (3)
      • ElasticSearch (2)
      • Kafka (4)
      • Spring (22)
        • Spring Cloud (7)
        • Security6 (5)
        • JPA (12)
        • 프로젝트 리팩토링 회고록 (4)
        • Logging (8)
        • Batch (2)
      • Redis (17)
        • Redis 개념 (8)
        • Redis 채팅 (5)
        • Redis 읽기쓰기 전략 (1)
      • AWS (11)
      • 리눅스 (29)
        • 리눅스 마스터 2급 (5)
        • 네트워크(기초) (7)
        • 리눅스의 이해 (6)
        • 리눅스의 설치 (2)
        • 리눅스 운영 및 관리 (6)
      • JAVA-기초 (16)
        • JAVA기본 (11)
        • Design Pattern (5)
      • JSP (27)
        • JSP 기본 개념 (10)
        • JSP (1)
      • SQL (1)
      • TIL (36)
      • 문제 풀이 (2)
        • Programmers (9)
        • 백준 문제풀이 (28)
      • JavaScript (10)
      • HTML (17)
      • Ngrinder (1)
        • Ngrinder 문서 정리 (1)
  • 블로그 메뉴

    • 링크

    • 공지사항

    • 인기 글

    • 태그

      CSS
      spring redis
      리눅스마스터2급정리
      JS
      자바 면접질문
      jsp request
      Spring Data Redis
      자바
      자바 알고리즘
      리눅스
      springsecurity
      리눅스마스터2급
      프로그래머스
      Springframework
      레디스
      자바스크립트
      Til
      스프링프레임워크
      jsp기초
      redis 채팅
      자바기초
      HTML
      java
      Spring
      자바 반복문
      redis
      백준
      JSP
      JavaScript
      자바 면접
    • 최근 댓글

    • 최근 글

    • hELLO· Designed By정상우.v4.10.3
    공부하고 기억하는 공간
    [Logging] 로그 출력을 효율적으로 생성 및 추적하기 7편 - Spring AOP
    상단으로

    티스토리툴바