[ AOP :Aspect Oriented Programming ]
: 관점 지향형 프로그램
코드 분리!
= 핵심 로직과 선택 로직의 분리!!
스프링 애플리케이션 대부분 특별한 경우를 제외하고 MVC 구조에서
Web Layer, Business Layer, Data Layer 세가지로 정의가 된다.
Web Layer
|
REST API 제공, Client 중심으로 적용
|
|
Business Layer
|
요청과 응답 등 내부 정책에 따라서 logic 개발
|
|
Data Layer
|
repository 불리는 데이터 베이스 및 외부와 연동을 처리 한다.
|
어떤 로직을 기준으로 핵심적인 관점, 부가적인 관점으로 나누어서 보고 그 관점을 기준으로 각각 모듈화하겠다는 것이다. 여기서 모듈화란 어떤 공통된 로직이나 기능을 하나의 단위로 묶는 것을 말한다.
예로들어 핵심적인 관점은 결국 우리가 적용하고자 하는 핵심 비즈니스 로직이 된다. 또한 부가적인 관점은 핵심 로직을 실행하기 위해서 행해지는 데이터베이스 연결, 로깅, 파일 입출력 등을 예로 들 수 있다.
AOP에서 각 관점을 기준으로 로직을 모듈화한다는 것은 코드들을 부분적으로 나누어서 모듈화하겠다는 의미다. 이때, 소스 코드상에서 다른 부분에 계속 반복해서 쓰는 코드들을 발견할 수 있는 데 이것을 흩어진 관심사 (Crosscutting Concerns)라 부른다.
AOP 주요 개념
- Aspect : 위에서 설명한 흩어진 관심사를 모듈화 한 것. 주로 부가기능을 모듈화함.
- Target : Aspect를 적용하는 곳 (클래스, 메서드 .. )
- Advice : 실질적으로 어떤 일을 해야할 지에 대한 것, 실질적인 부가기능을 담은 구현체
- JointPoint : Advice가 적용될 위치, 끼어들 수 있는 지점. 메서드 진입 지점, 생성자 호출 시점, 필드에서 값을 꺼내올 때 등 다양한 시점에 적용가능
- PointCut : JointPoint의 상세한 스펙을 정의한 것. 'A란 메서드의 진입 시점에 호출할 것'과 같이 더욱 구체적으로 Advice가 실행될 지점을 정할 수 있음
스프링 AOP 특징
- 프록시 패턴 기반의 AOP 구현체, 프록시 객체를 쓰는 이유는 접근 제어 및 부가기능을 추가하기 위해서임
- 스프링 빈에만 AOP를 적용 가능
- 모든 AOP 기능을 제공하는 것이 아닌 스프링 IoC와 연동하여 엔터프라이즈 애플리케이션에서 가장 흔한 문제(중복코드, 프록시 클래스 작성의 번거로움, 객체들 간 관계 복잡도 증가 ...)에 대한 해결책을 지원하는 것이 목적
즉 AOP 특정 부분에서 반복되는 Logic 들을 한 곳으로 몰아서
코딩을 할 수 있게 해주는 개념이다.
AOP 사용하는 방법
Refresh Gradle Project까지 눌러주면 aop를 사용이 가능하다
User Dto
package com.example.demo.dto;
import lombok.Data;
import lombok.ToString;
@Data
@ToString
public class User {
private Long id;
private String email;
private String pw;
}
Api Controller
package com.example.demo.controller;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import com.example.demo.dto.User;
import lombok.extern.slf4j.Slf4j;
@Slf4j
@RestController
@RequestMapping("/api")
public class ApiController {
// http://localhost:8080/api/get/[id값]?name=[value값]
@GetMapping("/get/{id}")
public String get(@PathVariable Long id, @RequestParam String name) {
return "id: " + id +", name: " + name ;
}
@PostMapping("/post")
public ResponseEntity<User> post(@RequestBody User user) {
return ResponseEntity.status(HttpStatus.OK).body(user);
}
}
ParameterAop
package com.example.demo.aop;
import java.lang.reflect.Method;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.stereotype.Component;
import lombok.extern.slf4j.Slf4j;
@Slf4j //로그 찍는 어노테이션
@Aspect
@Component
public class ParameterAop {
//controller 패키지 하위에 있는 모든 메소드를 찾아 보겠다 (관심을 갖겠다)
@Pointcut (value = "execution(* com.example.demo.controller..*.*(..))")
public void myPointCut() {}
// 메소드가 실행하기 이전
@Before ("myPointCut()")
public void myBefore(JoinPoint joinPoint) {
// aop를 활용해서 어떤 클래스에 어떤 메서드가 동작했는지 알아 볼 수 있다.
// 1. 메소드 이름을 알아보자
MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
//System.out.println("리턴 타입, 실행 된 풀 패키지 메소드 명 : " + methodSignature);
Method method = methodSignature.getMethod();
System.out.println("----------------------------------");
log.info("method - {}", method);
Object[] args = joinPoint.getArgs();
for (Object object : args) {
System.out.println("type : "+object.getClass().getSimpleName());
System.out.println("value : "+object);
}
System.out.println("----------------------------------");
}
@AfterReturning(value = "myPointCut()", returning = "returnObj")
public void myAfterReturn(JoinPoint joinPoint, Object returnObj) {
//매개변수로 들어오는 Object의 변수명은 returning 파라미터이름과 같아야 한다.
System.out.println("Return Value : " + returnObj);
}
}
'개발일지 > 스프링' 카테고리의 다른 글
SpringBoot - 스프링 개념 정의 (JPA) , 스프링 동작 원리 (0) | 2022.12.07 |
---|---|
SpringBoot 어노테이션 만들어 AOP 활용하기 (0) | 2022.12.07 |
@ExceptionHandler(어드바이스 - Advice) (0) | 2022.12.06 |
Spring Boot validation 활용 1 (0) | 2022.12.06 |
IoC - Inversion Of Control (0) | 2022.12.04 |
댓글