https://engkimbs.tistory.com/746
https://docs.spring.io/spring-framework/docs/2.5.x/reference/aop.html
AOP : Aspect Oriented Programming
AOP은 관점 지향 프로그래밍이라고 불린다. 관점지향은 쉽게 말해 어떤 로직을 기준으로 핵심적인 관점, 부가적인 관점으로 나누어서 보고 그 관점을 기준으로 각각 모듈화하겠다는 것이다. 여기서 모듈화란 어떤 공통된 로직이나 기능을 하나의 단위로 묶는것을 말한다.
예를 들어 핵심적인 관점은 결국 우리가 적용하고자 하는 핵심 비즈니스 로직이 되고 부가적인 관점은 핵심 로직을 실행하기 위해서 행해지는 데이터베이스 연결, 로깅, 파일 입출력 등을 예로 들 수 있다.
AOP에서 각 관점을 기준으로 로직을 모듈화한다는 것은 코드들을 부분적으로 나누어서 모듈화하겠다는 의미이다. 이 때, 소스 코드 사이에서 다른 부분에 계속 반복해서 쓰는 코드들을 발견할 수 있는데 이것을 흩어진 관심사(Crosscutting Concerns) 라 부른다.
위와 같이 흩어진 관심사를 Aspect로 모듈화하고 핵심적인 비즈니스 로직에서 분리하여 재사용하겠다는 것이 AOP의 목적이다.
A란 메서드의 진입 시점에 호출 할 것
과 같이 더욱 구체적으로 Adivce가 실행될 지점을 정할 수 있다.스프링
@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
스프링 빈으로 등록한다.
@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
를 적용하겠다는 의미이다.
public interface EventService {
void createEvent();
void publishEvent();
void deleteEvent();
}
@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");
}
}
@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();
}
}
Created an event
1033
Published an event
1005
Delete an event
0
또한 경로지정 방식 말고 특정 어노테이션이 붙은 포인트에 해당 Aspect를 실행할 수 있는 기능도 제공한다.
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.CLASS)
public @interface TestLogging {
}
@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;
}
@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");
}
}
Created an event
1041
Published an event
Delete an event
위 출력 결과에서 @TestLogging
어노테이션이 붙은 메서드만 Aspect가 적용된 것을 볼 수 있다.
마찬가지로 스프링 빈의 모든 메서드에 적용할 수 있는 기능도 제공한다.
@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;
}
Created an event
1048
Published an event
1004
Delete an event
0
위의 출력결과로 eventServiceImpl의 모든 메서드에 해당 Aspect가 추가된것을 알 수 있다.
이 밖에도 @Around
외에 타겟 메서드의 Aspect 실행 시점을 지정할 수 있는 애노테이션이 있다.
@ | 기능 |
---|---|
@Before(이전) | 어드바이스 타겟 메서드가 호출되기 전에 어드바이스 기능을 수행 |
@After(이후) | 타겟 메서드의 결과에 관계없이(성공,예외 관계 없이) 타겟 메소드가 완료되면 어드바이스 기능을 수행 |
@AfterReturning(정상적 반환 이후) | 타겟 메서드가 성공적으로 결과값을 반환 후에 어드바이스 기능을 수행 |
@AfterThrowing(예외 발생 이후) | 타겟 메소드가 수행 중 예외를 던지게 되면 어드바이스 기능을 수행 |
@Around(메소드 실행 전후) | 어드바이스가 타겟 메서드를 감싸서 타겟 메서드 호출전과 후에 어드바이스 기능을 수행 |