Notice
Recent Posts
Recent Comments
Link
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
Tags
- 그리디
- QueryDSL
- http
- 자바
- Android
- Proxy
- db
- Thymeleaf
- AOP
- 스프링
- transaction
- kotlin
- Servlet
- Exception
- pointcut
- spring
- JDBC
- Spring Boot
- 스프링 핵심 원리
- jpa
- 알고리즘
- SpringBoot
- 백준
- 인프런
- springdatajpa
- 김영한
- JPQL
- 스프링 핵심 기능
- java
- Greedy
Archives
- Today
- Total
개발자되기 프로젝트
[스프링AOP 포인트컷] @target, @within 본문
1. 정의
- @target : 실행 객체의 클래스에 주어진 타입의 애노테이션이 있는 조인 포인트
- @within : 주어진 애노테이션이 있는 타입 내 조인 포인트
2.설명
- @target , @within 은 다음과 같이 타입에 있는 애노테이션으로 AOP 적용 여부를 판단한다.
- @target(hello.aop.member.annotation.ClassAop) -->@ClassAop붙은 타입에 AOP 적용됨
- @within(hello.aop.member.annotation.ClassAop) -->@ClassAop붙은 타입에 AOP 적용됨
@ClassAop
class Target{}
3. @target vs @within
- @target
- 인스턴스의 모든 메서드를 조인 포인트로 적용한다.
- 부모 클래스의 메서드까지 어드바이스를 다 적용
- @within
- 해당 타입 내에 있는 메서드만 조인 포인트로 적용한다
- 자기 자신의 클래스에 정의된 메서드에만 어드바이스를 적용
4. Test
@Slf4j
@Import({AtTargetAtWithinTest.Config.class})
@SpringBootTest
class AtTargetAtWithinTest {
@Autowired
Child child;
@Test
void success() {
log.info("child Proxy={}", child.getClass());
child.childMethod(); //부모, 자식 모두 있는 메서드
child.parentMethod(); //부모 클래스만 있는 메서드
}
static class Config {
@Bean
public Parent parent() {
return new Parent();
}
@Bean
public Child child() {
return new Child();
}
@Bean
public AtTargetAtWithinAspect atTargetAtWithinAspect() {
return new AtTargetAtWithinAspect();
}
}
static class Parent {
public void parentMethod(){} //부모에만 있는 메서드
}
@ClassAop
static class Child extends Parent {
public void childMethod(){}
}
@Slf4j
@Aspect
static class AtTargetAtWithinAspect {
//@target: 인스턴스 기준으로 모든 메서드의 조인 포인트를 선정, 부모 타입의 메서드도 적용
@Around("execution(* hello.aop..*(..)) && @target(hello.aop.member.annotation.ClassAop)")
public Object atTarget(ProceedingJoinPoint joinPoint) throws Throwable {
log.info("[@target] {}", joinPoint.getSignature());
return joinPoint.proceed();
}
//@within: 선택된 클래스 내부에 있는 메서드만 조인 포인트로 선정, 부모 타입의 메서드는적용되지 않음
@Around("execution(* hello.aop..*(..)) && @within(hello.aop.member.annotation.ClassAop)")
public Object atWithin(ProceedingJoinPoint joinPoint) throws Throwable {
log.info("[@within] {}", joinPoint.getSignature());
return joinPoint.proceed();
}
}
}
[@target] void hello.aop.pointcut.AtTargetAtWithinTest$Child.childMethod()
[@within] void hello.aop.pointcut.AtTargetAtWithinTest$Child.childMethod()
[@target] void hello.aop.pointcut.AtTargetAtWithinTest$Parent.parentMethod()
- parentMethod() 는 Parent 클래스에만 정의되어 있고,
- Child 클래스에 정의되어 있지 않기 때문에 @within 에서 AOP 적용 대상이 되지 않는다.
- 실행결과를 보면 child.parentMethod() 를 호출 했을 때 [@within] 이 호출되지 않음
5. 주의
- 다음 포인트컷 지시자는 단독으로 사용하면 안된다. args, @args, @target
- 이번 예제를 보면 execution(* hello.aop..*(..)) 를 통해 적용 대상을 줄여준 것을 확인할 수 있다.
- args , @args , @target 은 실제 객체 인스턴스가 생성되고 실행될 때 어드바이스 적용 여부를 확인할 수 있다.
- 실행 시점에 일어나는 포인트컷 적용 여부도 결국 프록시가 있어야 실행 시점에 판단할 수 있다.
- 프록시가 없다면 판단 자체가 불가능하다.
- 그런데 스프링 컨테이너가 프록시를 생성하는 시점은
스프링 컨테이너가 만들어지는 애플리케이션 로딩 시점에 적용할 수 있다. - 따라서 args , @args , @target 같은 포인트컷 지시자가 있으면 스프링은 모든 스프링 빈에 AOP를 적용을 시도.
- 앞서 설명한 것 처럼 프록시가 없으면 실행 시점에 판단 자체가 불가능하다.
따라서 실행 시점에 AOP적용 판단을 위해 스프링 입장에서는 모든 빈에 프록시 적용이 필요함. - 문제는 이렇게 모든 스프링 빈에 AOP 프록시를 적용하려고 하면 스프링이 내부에서 사용하는 빈 중에는
final 로 지정된 빈들도 있기 때문에 오류가 발생할 수 있다. - 따라서 이러한 표현식은 최대한 프록시 적용 대상을 축소하는 표현식과 함께 사용해야 한다.
6. GitHub: 220107 @target, @within
'인프런 > [인프런] 스프링 핵심 원리 - 고급' 카테고리의 다른 글
[스프링AOP 포인트컷] bean (0) | 2022.01.07 |
---|---|
[스프링AOP 포인트컷] @annotation, @args (0) | 2022.01.07 |
[스프링AOP 포인트컷] args (0) | 2022.01.07 |
[스프링AOP 포인트컷] within (0) | 2022.01.06 |
[스프링AOP 포인트컷] execution - 2 (0) | 2022.01.06 |
Comments