프로그램/Spring 3.0
@Aspect 어노테이션을 이용한 AOP - @Pointcut 어노테이션을 이용한 Pointcut 설정
영혼과영혼의양식
2012. 5. 11. 21:03
●XML 스키마를 사용할 때에는 <aop:pointcut> 태그와 각 Advice 설정 태그의 pointcut-ref 속성을 이용해서 Pointcut을 설정하고 여러 Advice에서 동일한 Pointcut을 참조할 수 있음
=> 게시물 [Advice 타입 별 클래스 작성 - 여러 Advice에 동일한 Pointcut 참조] 참고
●@Aspect 어노테이션을 사용하는 경우에는 @Pointcut 어노테이션을 이용해서 Pointcut 설정을 재사용할 수 있음.
FaqBoardImpl.java
package sp.aop.service;
import org.springframework.stereotype.Component;
@Component("faqBoard") public class FaqBoardImpl implements Board {
@Override public String getBoardName() { // TODO Auto-generated method stub return "FAQ Board"; }
@Override public String checkBoard() { // TODO Auto-generated method stub return "N"; } } |
NoticeBoardImpl.java
package sp.aop.service;
import org.springframework.stereotype.Component;
@Component("noticeBoard") public class NoticeBoardImpl implements Board {
@Override public String getBoardName() { // TODO Auto-generated method stub // int num = 100 / 0; return "Notice Board"; }
@Override public String checkBoard() { // TODO Auto-generated method stub return "Y"; } } |
BoardController.java
package sp.aop.controller;
import javax.annotation.Resource; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.ModelAndView;
import sp.aop.service.Board;
public class BoardController extends org.springframework.web.servlet.mvc.AbstractController {
@Resource(name="noticeBoard") private Board board; @Override protected ModelAndView handleRequestInternal(HttpServletRequest arg0, HttpServletResponse arg1) throws Exception { // TODO Auto-generated method stub System.out.println("<<===== START =====>>"); System.out.println("Board Name-1 : " + board.getBoardName()); System.out.println(""); System.out.println("Board Check-1 : " + board.checkBoard()); System.out.println("<<===== END =====>>"); return null; }
} |
TestAspect.java
package sp.aop.aspect;
import org.aspectj.lang.JoinPoint; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.AfterReturning; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;
@Aspect public class TestAspect { @Pointcut("execution(public * sp.aop.service.*.*())") private void allTarget(){};
@Before("allTarget()") public void aaa(JoinPoint joinPoint){ System.out.println("--- before ---"); } //pointcut을 aspectJ의 표현식으로 작성 혹은 @Pointcut 어노테이션이 적용된 메서드 이름을 작성 @AfterReturning(pointcut="sp.aop.aspect.TestAspect.allTarget()", returning="ret") public void bbb(JoinPoint joinPoint, Object ret){ System.out.println("--- after returning : " + ret.toString() + " ---"); } @After("allTarget()") public void ddd(JoinPoint joinPoint){ System.out.println("--- after ---"); } @Around("allTarget()") public Object eee(ProceedingJoinPoint joinPoint) throws Throwable{ Object obj = joinPoint.proceed(); System.out.println("--- around ---"); return obj; } }
|
●@Pointcut 어노테이션은 Pointcut 표현식을 값으로 가지며 @Pointcut 어노테이션이 적용된 메서드는 리턴 타입이 void
●@Pointcut 어노테이션이 적용된 메서드는 메서드 몸체에 코드를 갖지 않음(코드를 가져도 의미가 없음)
●@Pointcut 어노테이션을 이용해서 Pointcut을 정의하면, Advice 관련 어노테이션에서 @Pointcut 어노테이션이 적용된 메서드(@Pointcut 메서드)의 이름을 이용해서 Pointcut을 참조
메서드 이름은 다음과 같이 범위에 따라서 알맞게 입력해야 함
■ 같은 클래스에 위치한 @Pointcut 메서드는 '메서드이름'만 입력
■ 같은 패키지에 위치한 @Pointcut 메서드는 '클래스단순이름.메서드이름'을 입력
■ 다른 패키지에 위치한 @Pointcut 메서드는 '완전한클래스이름.메서드이름'을 입력
●@Pointcut 메서드 사용시 주의할 점은 메서드 접근자를 주의해서 사용 => private으로 설정된 @Pointcut 메서드는 다른 클래스에서 사용할 경우 예외가 발생(기본적으로 알고 있는 접근 제어자와 같음)
spring302-servlet.xml
결과
<<===== START =====>> --- before --- --- after returning : Notice Board --- --- after --- --- around --- Board Name-1 : Notice Board
--- before --- --- after returning : Y --- --- after --- --- around --- Board Check-1 : Y <<===== END =====>> |
[참고자료] Spring 3.0 프로그래밍-최범균