AOP (Aspect Oriented Programming) 2015 Web Service Computing
트랜잭션의 구현 만약 @Transactional을 직접 구현한다면? 비즈니스 로직 코드와 트랜잭션 관련 코드가 명백하게 분리되어 있긴 하지만 트랜잭션 관련 코드가 비즈니스 로직 코드와 함께 공존 모든 DAO 클래스에 적용한다면? 동일한 기능의 중복 불가피
AOP란 무엇인가? 앞서 트랜잭션과 같은 부가기능에 해당하는 기능의 모듈화 작업은 기존의 객체지향 설계 패러다임과 구분되는 새로운 특성이 있다고 판단 애스펙트 (Aspect) 객체 지향 기술에서 부가기능 모듈을 부르는 이름 애스펙트 자체로는 애플리케이션의 핵심기능을 담고 있지는 않지만 요소요소마다 공통 관심사항이 될 수 있음 즉, 애플리케이션을 구성하는 한가지 측면
AOP란 무엇인가? 애스펙트 (Aspect)의 분리 왼쪽 오른쪽 부가기능이 핵심기능의 모듈에 침투해 들어가면서 설계와 코드가 지저분해짐 부가기능 코드가 여기저기 메소드에 마구 흩어져서 나타나고 코드는 중복됨 오른쪽 코드 사이에 침투한 부가기능을 독립적인 모듈인 애스팩트로 구분 각각 성격이 다른 부가기능은 다른 면에 존재하여 독립적으로 그 코드를 살펴 볼 수 있음
AOP란 무엇인가? 애스펙트 지향 프로그래밍 (Aspect Oriented Programming) 관점 지향 프로그래밍 애플리케이션의 핵심적인 기능에서 부가적인 기능을 분리하여 애스펙트라는 모듈로 구분하여 설계하고 개발하는 방법 AOP는 OOP를 돕는 보조적인 기술이지 OOP를 대체하는 기술은 아님 애플리케이션을 다양한 측면에서 독립적으로 모델링하고, 설계하고, 개발할 수 있도록 함
AOP 용어 타겟 (Target) 어드바이스 (Advice) 조인 포인트 (Join Point) 포인트컷 (Pointcut) 부가기능을 부여할 대상 (예: 클래스) 어드바이스 (Advice) 타겟에게 제공할 부가기능을 담은 모듈 객체로 정의하기도 하지만 메소드 레벨에서 정의하기도 함 조인 포인트 (Join Point) 어드바이스가 적용될 수 있는 위치 스프링 프록시 AOP에서 조인 포인트는 메소드의 실행 단계임 포인트컷 (Pointcut) 어드바이스를 적용할 조인 포인트를 선별하는 작업 또는 그 기능을 정의한 모듈 스프링의 포인트컷은 특정 타입 객체의 특정 메소드를 선정하는 기능
AOP 용어 프록시 (Proxy) 어드바이저 (Advisor) 위빙 클라이언트와 타깃 사이에 투명하게 존재하여 부가기능을 제공하는 객체 DI를 통해 타겟 대신 클라이언트에 주입 어드바이저 (Advisor) 포인트컷과 어드바이스를 지닌 객체 즉, 어떤 부가기능(어드바이스)을 어디에(포인트컷) 전달할 것인가를 알고 있는 객체 스프링의 어드바이저는 아주 단순한 애스펙트의 한 형태 위빙 어드바이스를 포인트컷에 전달하는 과정
@AspectJ를 이용한 AOP 적용 @AspectJ란 AOP를 활용하여 실습하고자 하는 것 @AspectJ는 Java Annotation을 사용한 일반 Java 클래스로 관점(Aspect)를 정의하는 방식이다. @AspectJ 방식은 AspectJ 5 버전에서 소개되었으며, Spring은 2.0 버전부터 AspectJ 5 어노테이션을 지원한다. AOP를 활용하여 실습하고자 하는 것 1) 앞서 만들었던 회원가입 기능(단순 DB 삽입)을 서비스(AOP 타겟)로 구현 2) 위 서비스의 특정 함수를 실행할 때 실행 전 후로 부가기능(간단한 문구 출력) 삽입
AOP 실습을 위한 구성 pom.xml에 다음과 같이 추가 <!-- Spring AOP + AspectJ --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aop</artifactId> <version>4.2.0.RELEASE</version> </dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjrt</artifactId> <version>1.8.0</version> <artifactId>aspectjweaver</artifactId> <version>1.8.6</version>
AOP 실습을 위한 구성 koreatech.cse.service.UserService package koreatech.cse.service; import koreatech.cse.domain.User; import koreatech.cse.repository.UserMapper; import org.springframework.stereotype.Service; import javax.inject.Inject; import java.util.Date; @Service public class UserService { //타겟 @Inject private UserMapper userMapper; public Boolean signup(User user) { if(user.getEmail() == null || user.getPassword() == null) return false; userMapper.insert(user); System.out.println(“User created: " + new Date()); return true; }
AOP 실습을 위한 구성 koreatech.cse.aop.AopTester 스프링 설정파일에 다음과 같이 추가 package koreatech.cse.aop; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import java.util.Date; @Aspect public class AopTester { @Before("execution(* koreatech.cse.service.UserService.signup(..))") //포인트컷 public void beforeMethod(JoinPoint joinPoint) { //어드바이스 System.out.println("Before " + joinPoint.getSignature().getName() + ": " + new Date()); } @After("execution(* koreatech.cse.service.UserService.signup(..))") public void afterMethod(JoinPoint joinPoint) { System.out.println("After " + joinPoint.getSignature().getName() + ": " + new Date()); 스프링 설정파일에 다음과 같이 추가 <aop:aspectj-autoproxy /> <bean id="userServiceAspect" class="koreatech.cse.aop.AopTester" />
AOP 실습을 위한 구성 UserController를 다음과 같이 수정 실행 결과 @Transactional @RequestMapping(value="/signup", method= RequestMethod.POST) @ResponseBody public String signup(@ModelAttribute User user) { userService.signup(user); return "success"; } 실행 결과 Before: Fri Sep 11 14:36:41 JST 2015 User created at Fri Sep 11 14:36:41 JST 2015 After: Fri Sep 11 14:36:41 JST 2015
@AfterReturning Pointcut에 지정한 함수가 실행 완료 후 결과값을 활용 @AfterReturning(pointcut = "execution(* koreatech.cse.service.UserService.signup(..))", returning= "result") public void afterReturningMethod(JoinPoint joinPoint, Object result) { System.out.println("After Returning " + joinPoint.getSignature().getName() + ": " + new Date() + ", Value = " + result); }
@Around @Before와 @After를 통합하여 실행 가능 @Around("execution(* koreatech.cse.service.UserService.signup(..))") public void aroundMethod(ProceedingJoinPoint joinPoint) throws Throwable { System.out.println("@Around is running!"); System.out.println("Before " + joinPoint.getSignature().getName()); joinPoint.proceed(); System.out.println("After " + joinPoint.getSignature().getName()); }
(참고) 포인트컷 표현식 보다 복잡하고 세밀한 포인트컷 필터 기준 설정 포인트컷 표현식 클래스와 메소드 이름, 정의된 패키지, 메소드 파라미터, 리턴 값, 부여된 어노테이션, 구현한 인터페이스, 상속한 클래스 포인트컷 표현식 효과적으로 포인트컷의 클래스와 메소드는 선정하는 언어 문법은 다음과 같다.
(참고) 포인트컷 표현식 포인트컷 표현식 테스트용 인터페이스 포인트컷 표현식 테스트용 클래스 1
(참고) 포인트컷 표현식 포인트컷 표현식 테스트용 클래스 2 클래스 선정 기능 테스트용
표현식과 선정 결과
표현식과 선정 결과