스프링 AOP 기본 개념

23896 단어 SpringSpring

출처 및 참고

https://engkimbs.tistory.com/746
https://docs.spring.io/spring-framework/docs/2.5.x/reference/aop.html


스프링 AOP

AOP : Aspect Oriented Programming

AOP은 관점 지향 프로그래밍이라고 불린다. 관점지향은 쉽게 말해 어떤 로직을 기준으로 핵심적인 관점, 부가적인 관점으로 나누어서 보고 그 관점을 기준으로 각각 모듈화하겠다는 것이다. 여기서 모듈화란 어떤 공통된 로직이나 기능을 하나의 단위로 묶는것을 말한다.

예를 들어 핵심적인 관점은 결국 우리가 적용하고자 하는 핵심 비즈니스 로직이 되고 부가적인 관점은 핵심 로직을 실행하기 위해서 행해지는 데이터베이스 연결, 로깅, 파일 입출력 등을 예로 들 수 있다.

AOP에서 각 관점을 기준으로 로직을 모듈화한다는 것은 코드들을 부분적으로 나누어서 모듈화하겠다는 의미이다. 이 때, 소스 코드 사이에서 다른 부분에 계속 반복해서 쓰는 코드들을 발견할 수 있는데 이것을 흩어진 관심사(Crosscutting Concerns) 라 부른다.

위와 같이 흩어진 관심사를 Aspect로 모듈화하고 핵심적인 비즈니스 로직에서 분리하여 재사용하겠다는 것이 AOP의 목적이다.


AOP 주요 개념

  • Aspect
    • 위에서 설명한 흩어진 관심사를 모듈화한것
    • 주로 부가기능을 모듈화한다.
  • Target
    • Aspect를 적용하는 곳(클래스,메서드..)
  • Advice
    • 실질적으로 어떤 일을 해야 할지에 대한 것
    • 실질적인 부가기능을 담은 구현체
  • JoinPoint
    • Advice가 적용될 위치, 끼어들 수 있는 지점, 메서드 진입 지점, 생성자 호출 시점, 필드에서 값을 꺼내올 때 등 다양한 시점에 적용가능
  • PointCut
    • JoinPoint의 상세한 스펙을 정의한 것이다.
    • A란 메서드의 진입 시점에 호출 할 것과 같이 더욱 구체적으로 Adivce가 실행될 지점을 정할 수 있다.

AOP 특징

  • 프록시 패턴 기반의 AOP 구현체
    • 프록시 객체를 쓰는 이유는 접근 제어 및 부가기능을 추가하기 위해서이다.
  • 스프링 빈에만 AOP 적용 가능
  • 모든 AOP 기능을 제공하는 것이 아닌 스프링 IoC와 연동하여 엔터프라이즈 애플리케이션에 가장 흔한 문제(중복코드,프록시 클래스 작성의 번거로움,객체들 간 결합도 증가..)에 대한 해결책을 지원하는 한다.

스프링 AOP : @AOP

스프링 @AOP를 사용하기 위해서는 다음과 같은 의존성을 추가해야 한다.

// https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-aop
	implementation 'org.springframework.boot:spring-boot-starter-aop:2.6.2'

다음에는 아래와 같이 @Aspect 애노테이션을 붙여 이 클래스가 Aspect를 나타내는 클래스라는 것을 명시하고 @Component 스프링 빈으로 등록한다.

TestAspect

@Aspect
@Component
public class TestAspect {

    @Around("execution(* me.dragonappear..*.EventService.*(..))")
    public Object log(ProceedingJoinPoint pjp) throws Throwable {
        long begin = System.currentTimeMillis();
        Object retVal = pjp.proceed(); // 메서드 호출 자체를 감싼다
        System.out.println(System.currentTimeMillis() - begin);
        return retVal;
    }
}

@Around 애노테이션은 타켓 메서드를 감싸서 특정 Advice를 실행한다는 의미이다. 위 코드의 Advice는 타켓 메서드가 실행된 시간을 측정하기 위한 로직을 구현하였다. 추가적으로 execution(*me.dragonappear..*.EventService.*(..))가 의미하는 것은 me.dragonappear 아래의 패키지 경로의 EventService 객체의 모든 메서드에 이 Aspect를 적용하겠다는 의미이다.

EventService

public interface EventService {
    void createEvent();

    void publishEvent();

    void deleteEvent();
}

EventServiceImpl

@Component
public class EventServiceImpl implements EventService{

    @Override
    public void createEvent() {
        try {
            Thread.sleep(1000);
        } catch(InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Created an event");
    }

    @Override
    public void publishEvent() {
        try {
            Thread.sleep(1000);
        } catch(InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Published an event");
    }

    @Override
    public void deleteEvent() {
        System.out.println("Delete an event");
    }
}

AppRunner

@RequiredArgsConstructor
@Service
public class AppRunner implements ApplicationRunner {
    private final EventService eventService;

    @Override
    public void run(ApplicationArguments args) throws Exception {
        eventService.createEvent();
        eventService.publishEvent();
        eventService.deleteEvent();
    }
}

Result

Created an event
1033
Published an event
1005
Delete an event
0

또한 경로지정 방식 말고 특정 어노테이션이 붙은 포인트에 해당 Aspect를 실행할 수 있는 기능도 제공한다.

TestLogging

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.CLASS)
public @interface TestLogging {
}

TestAspect

@Around("@annotation(TestLogging)")
    public Object log(ProceedingJoinPoint pjp) throws Throwable {
        long begin = System.currentTimeMillis();
        Object retVal = pjp.proceed(); // 메서드 호출 자체를 감싼다
        System.out.println(System.currentTimeMillis() - begin);
        return retVal;
    }

EventServiceImpl

@Component
public class EventServiceImpl implements EventService{

    @TestLogging
    @Override
    public void createEvent() {
        try {
            Thread.sleep(1000);
        } catch(InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Created an event");
    }

    @Override
    public void publishEvent() {
        try {
            Thread.sleep(1000);
        } catch(InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Published an event");
    }

    @Override
    public void deleteEvent() {
        System.out.println("Delete an event");
    }
}

Result

Created an event
1041
Published an event
Delete an event

위 출력 결과에서 @TestLogging 어노테이션이 붙은 메서드만 Aspect가 적용된 것을 볼 수 있다.

마찬가지로 스프링 빈의 모든 메서드에 적용할 수 있는 기능도 제공한다.

TestAspect

@Around("bean(eventServiceImpl)")
    public Object log(ProceedingJoinPoint pjp) throws Throwable {
        long begin = System.currentTimeMillis();
        Object retVal = pjp.proceed(); // 메서드 호출 자체를 감싼다
        System.out.println(System.currentTimeMillis() - begin);
        return retVal;
    }

Result

Created an event
1048
Published an event
1004
Delete an event
0

위의 출력결과로 eventServiceImpl의 모든 메서드에 해당 Aspect가 추가된것을 알 수 있다.

이 밖에도 @Around외에 타겟 메서드의 Aspect 실행 시점을 지정할 수 있는 애노테이션이 있다.

@기능
@Before(이전)어드바이스 타겟 메서드가 호출되기 전에 어드바이스 기능을 수행
@After(이후)타겟 메서드의 결과에 관계없이(성공,예외 관계 없이) 타겟 메소드가 완료되면 어드바이스 기능을 수행
@AfterReturning(정상적 반환 이후)타겟 메서드가 성공적으로 결과값을 반환 후에 어드바이스 기능을 수행
@AfterThrowing(예외 발생 이후)타겟 메소드가 수행 중 예외를 던지게 되면 어드바이스 기능을 수행
@Around(메소드 실행 전후)어드바이스가 타겟 메서드를 감싸서 타겟 메서드 호출전과 후에 어드바이스 기능을 수행

좋은 웹페이지 즐겨찾기