기본 콘텐츠로 건너뛰기

[Spring] AOP를 알아보자

이번 시간에는 AOP라는 것에 대해 다뤄보려한다.
실무에선 거의 사용되지 않는 것이지만 회사 면접 등에서 물어 볼 수 있으니 이론 위주로 설명하려한다.

인트로
구현하고자 하는 비즈니스 로직과 다소 거리가 있거나 여러 모듈에 걸쳐 공통적이고 반복적으로 필요로 하는
처리 내용을 횡단 관심사라 한다.
보안, 로깅, 트랜젝션관리, 모니터링, 캐시처리, 예외처리 등등이 이에 해당한다.
횡단 관심사에 해당 하는 부분을 분리해 한 곳으로 모으는 것을 ‘횡단 관심사의 분리' 라고 하고
이를 실행하는 방법을 ‘관점지향 프로그래밍'이라 한다.


AOP 개요
AOP를 활용하면 애플리케이션 코드에서 공통적인 기능을 분리해 낼 수 있다.


AOP 용어
  • Aspect : AOP의 단위가 되는 횡단 관심사에 해당. ex) 로그를 출력한다. 예외를 처리한다 등
  • JoinPoint : 횡단 관심사가 실행될 지점이나 시점. 스프링에선 메서드 단위로 조인포인트를 잡는다.
  • Advice : 특정 조인 포인트에서 실행되는 코드. 횡단 관심사를 실제 구현해 처리하는 부분.
  • Pointcut : 수많은 조인 포인트 중에 실제로 어드바이스를 적용할 곳을 선별하기 위한 표현식. 조인 포인트의 구룹.
  • Weaving : 애플리케이션 코드의 적절한 지점에 애스팩트를 적용하는 것을 말함.
  • Target  : AOP 처리에 의해 처리 흐름에 변화가 생긴 객체를 말함.


스프링에서 지원하는 어드바이스 유형
  • Before : 조인 포인트 전에 실행된다. 예외 발생하는 경우 제외하고 항상 실행됨.
  • After Returning : 조인 포인트가 정상적으로 종료한 후 실행.
  • After Throwing : 조인 포인트에서 예외가 발생했을 때 실행
  • After : 조인 포인트 처리 완료 후 실행.
  • Around : 조인 포인트 전 후에 실행


스프링 AOP
스프링 프레임 워크 안에 스프링 AOP가 포함돼 있다. 스프링 AOP에는 DI 컨테이너에서 관리하는 빈들을 타깃으로
어드바이스를 적용하는 기능이 있는데, 조인 포인트에 어드바이스를 적용하는 방법은 프락시 객체를 만들어
대체하는 방법을 쓴다.
어드바이스가 적용된 이후 DI 컨테이너에서 빈을 꺼내면 프락시 형태로 어드바이스 기능이 덮혀진 빈이 나온다.


애스팩트를 동작시키려면 자바기반 설정방식에서 설정 클래스에 @CnableAspectJAutoProxy를 추가해야 한다.

@SpringBootApplication
@EnableAspectJAutoProxy
public class JboardApplication {

 @Bean
 public MyUtil myUtil() {
    return new MyUtil();
 }

 public static void main(String[] args) {
    SpringApplication.run(JboardApplication.class, args);
 }
}


XML 기반설정으로는 <aop:aspectj-autoproxy/>를 추가해 줘야 한다.


자바 기반 설정 방식에서의 어드바이스 정의
어드바이스 기능을 하는 메서드에 @Before 애너테이션을 붙인 다음, 포인트컷 표현식을 추가해 주면 된다.
메서드의 매개 변수로 JoinPoint를 선언하고 있는데, 매서드가 호출 될 때 전달 되는 인수를 통해
실행중인 매서드의 정보를 구할 수 있다.

@Aspect
@Component
public class MethodStartLoggingAspect {
  @Before("execution(* *..*ServiceImpl.*(..))")
  public void startLog(JoinPoint jp) {}
}
xml로도 설정하는 방법이 있다. xml 방식은 직접 다루진 않겟다.


스프링 프로젝트에서 활용되는 AOP 기능
1. 트랜잭션 관리
트랜젝션 관리가 필요한 메서드에 @Transactional 애너테이션을 지정하면 스프링 프레임워크가 대신
트랜젝션 관리를 해준다. 스프링 프레임워크에서 해당 메서드가 정상 종료한 것이 확인 되면 커밋 하고,
실패하면 롤백 한다.


2. 인가
권한 제어가 필요한 메서드에 @PreAuthrize를 지정하면 메서드 호출 전 특정 인가 조건을 만족하는지 확인할 수 있다. @PreAuthrize(“hasRole(‘ADMIN’)”) 메서드 사용자가 관리자인지 확인


3. 캐싱
메서드에 @Cacheable을 지정하면 메서드의 매개변수 등을 키로 사용해 메서드의 실행 결과를 캐시로 관리 할 수 있다. 캐시에 키가 있는 메서드를 호출하면 캐시에 등록된 반환값을 돌려준다. @Cachable(“user”)


4. 비동기 처리
비동기 처리가 하고 싶다면 @Async 애너테이션을 붙여준다. 이는 별도의 스래드에서 실행될 수 있다.
반환값은 CompletableFuture or DeferredResult 타입이어야 한다.


5. 재처리
@Retryable을 붙여주면 된다.


데이터 바인딩과 형변환

자바 객체의 프로퍼티에 외부에서 입력된 값을 설정하는 과정을 데이터 바인딩이라 한다.
데이터 바인딩 형변환이 없다면  웹 애플리케이션에서 요청 파라미터를 받아 올 때
HttpServletRequest 인터페이스의 getParameter 메서드를 사용해야 하는데,
여간 불편한 과정을 거처야 햔다. 스프링은 이런 형 변환을 제공한다.

이외 리소스 관리 메세지 관리 등등이 있지만 이번 절에선 여기까지 다뤄 보겠다

그럼 끝~!!

댓글

이 블로그의 인기 게시물

[스프링부트2.0 낚시게시판] 01. 프로젝트 생성 및 환경을 세팅해 보자

첫번째. 이클립스에서 프로젝트를 생성해 보자. 빠밤! 1. 이클립스 실행하고 프로젝트 생성하기  - 이클립스 실행 후 File -> New -> Spring Starter Project클릭 ( 부트는 Spring Starter Project로!! )        해당 프로젝트 설정을 본인의 입맛에 맞게? 해주자. 처음엔 저와 똑같이 하는게 삽질(?)의 노고를 덜 수 있으니 저 같은 초보 개발자 분들이나 이제 막 공부를 시작 하셧다면 위와 가이 설정 하는걸 추천.  - New Spring Starter Project Dependencies    - Spring Boot Version : 2.0.2    - Core : DevTools, Security, Lombok 클릭    - Web : web  클릭    - SQL : JPA, H2 클릭    - template Engines : Mstache   --> 타임리프(요즘 회사에서 많이 쓴다고해서)   - 그다음 다음 -> Finish 클릭하게 되면 Maven에 관한 프로젝트가 생성된다.   처음엔 메이븐 디펜던시부분을 받느라 시간이 걸릴수 있다. 프로젝트 구조는 스프링과 별반 차이가 없어 보인다 프로젝트 구조 관련해선 조만간 포스팅 해봐야 겠다. (요즘 책 읽을 시간도 없어서...😂) 두번째. 실행을 해보자.  스프링 부트2.0의 특징은 자체적으로 톰켓이 내장 되어 있어 따로 톰켓을 설정하는 부분이 없어서 아주 매우 편안하게 되었다.    - 실행은 src/main/java 밑에 com.fishing.board 패키지 밑에 FishBoardApplication.java 오른쪽 클릭 후...

[SpringBoot] Spring Data JPA를 알아보자 [2탄 엔티티를 다루기]

이번 시간에는 엔티티 클래스 설계서 부터 테스트까지 진행해 보도록 하겠다. 1. 엔티티 클래스 설계   - JPA는 자동으로 테이블을 생성하는 기능을 가질 수 있다. 다음 2가지 방법이 있다.     1) SQL을 이용해 테이블을 먼저 생성하고 엔티티 클래스를 만드는 방법     2) JPA를 이용해 클래스만 설계하고 자동으로 테이블을 생성하는 방법   - 이중에서 2)번 방법을 알아보자.   - JPA 엔티티 클래스를 생성하는 작업은 다음 과정을 거친다     1) 클래스 설계     2) 각종 애너테이션을 이용해 제약 조건 추가 설정     3) 엔티티 간 연관관계 설정 1.1 엔티티 클래스 설계    package com . example . sutdy . domain ; import lombok.Getter ; import lombok.Setter ; import lombok.ToString ; import java.sql.Timestamp ; @Getter @Setter @ToString public class Board { private Long bno; private String title; private String writer; private String content; private Timestamp regdate; private Timestamp updatedate; }   - 위와 같이 일반적인 방법으로 클래스를 만들어 봤다. 다음 JPA 어노테이션에 관해 알아보고 붙여보자 1.1.2 JPA 어노테이션   - @Id : 각 엔티티를 구별할수 있도록 식별 ID부과.(일종의 primary key로 보면된다). 모든 엔티티에 반드시 지정하자.   - @Column : 인스턴스 ...

[스프링부트2.0 낚시게시판] 03. 네비게이션화를 해보자

이번시간에는 네비게이션화 일명 페이지 요청당 Path에 대해 작성해보자 화면 HTTP Request method parameter 설명 메인화면 / GET 루어낚시 메인 /lure GET ?/location='seoul'&room='aa'&context='aa'&page=1 location(지역), room(낚시터명), context(검색어) 루어낚시 상세보기 /lure/{no} GET ?/location='seoul'&room='aa'&context='aa'&page=1&no=1 location(지역), room(낚시터명), context(검색어), no(글번호) 루어낚시 댓글달기 /lure/{no}/reply POST ajax로 낚시스쿨 메인 /school GET 낚시스쿨 상세보기 /school/{no} GET 낚시스쿨 댓글달기 /school/{no}/reply POST 중고장터 메인 /shop GET 공지사항 메인 /notice GET ?category_no=1&page=1 로그인 화면 /users/login GET 로그인 성공 시 전에 요청했던 화면으로 이동하게끔 로그인 /users/login POST userid=""&password=""&rememberme="" 아이디, 패스워드를 입력 받고, rememberme 기능을 bool값으로 회원가입 화면 /users/join GET 회원가입 /users/join POST userid=""&birth=""&sex=""&phone=""&password=""&email="" 위는 HTTP 요청에 따른 화면전환 방법을 ...