Yeon's Frame
  • Intro
  • Project
    • 미소웨더
      • 기획 - 아이디어 및 유스케이스 정리
      • DB 설계 - JPA 활용 중심
  • Study
    • Spring
      • Spring Feature
      • N+1
      • OSIV
      • @Valid
      • Spring Boot Feature
      • Spring 5 Feature
      • JPA vs MyBatis
      • Filter, Interceptor
      • Persistence Context
      • @Transactional
      • @Controlleradvice, @ExceptionHandler
      • Spring Security
      • Dispatcher Servlet
      • @EnableWebMvc
      • Stereo Type
      • AOP
      • JPA Repository
    • Infrastructure
      • Git
      • DNS
      • JWT
      • DevOps
      • Docker
      • Jenkins
      • Cloud Computing
      • MSA
    • Clean Code
      • 깨끗한 코드
      • 의미있는 이름
      • 함수
      • 주석
      • 형식 맞추기
      • 객체와 자료구조
      • 오류 처리
      • 경계
      • 단위테스트
      • 클래스
      • 시스템
  • Basic
    • Java
      • OOP
      • OOL Features
      • Class & Objects
      • Instance & Heap Memory
      • Constructor
      • Reference Type Uses
      • Access Modifier & Hiding
      • This
      • Collaboration
      • Static
      • Inheritance
      • Polimorphism & Casting
      • Abstract Class
      • Interface
      • Object Class
    • Java Coding
      • JVM
      • String, StringBuffer, StringBuilder
      • SOF
      • Blockcing/NonBlocking
      • Enum
      • Static
      • Thread
      • hashCode() | equals()
      • JDK8
      • Stream
      • Optional
      • Lambda & Closure
      • Exception
      • Garbage Collecter
      • Collection
      • Call by Value & Call by Reference
      • Generic
    • Java_Advance
      • Inner Class
      • Lambda Expression
      • Functional Interface
      • OOP vs Lambda Expression
      • Stream
      • Reduce()
      • Exception Handling
      • Custom Exception
      • Error Log
      • IO Stream
      • IO Stream - Serialization
      • File Class
      • Decorator Pattern
      • Thread
    • Data Structure
      • Intro
      • Generic(1)
      • Generic(2) T extends
      • Generic(3) Method
      • Collection Frameworks
      • List
      • Iterator
      • Set
      • Comparable & Comparator
      • Map
  • Practice
    • Algorithm
      • Strategy
        • Primitive, Reference, Generic
        • Number Data Types
        • For-Each
        • Array, Queue, ArrayDeque
        • Array vs ArrayList
Powered by GitBook
On this page
  • @Valid가 동작하는 원리를 설명하시오
  • Spring Web MVC 과정
  • HandlerMapping
  • RequestMappingHandlerAdaper
  • DispatcherServlet
  • @Valid, @Validated의 차이는 무엇인가?
  • 참고
  1. Study
  2. Spring

@Valid

@Valid 동작 원리

PreviousOSIVNextSpring Boot Feature

Last updated 2 years ago

@Valid가 동작하는 원리를 설명하시오

  • 우선 프로세스를 설명하기 전에 web mvc가 동작하는 원리부터 이해해야 한다.

Spring Web MVC 과정

  • 사용자의 요청이 오면 Filter를 통해서 DispatcherServlet을 호출한다.

  • DispatcherServlet은 어떤 컨트롤러를 호출해야 할지 HandlerMapping 과정을 통해서 찾아간다.

  • HandlerMapping을 확인하면 해당하는 HandlerMappingAdapter를 통해서 interceptor, argumentResolver를 과정을 순회한다. 그리고 그 이후에 Controller를 호출하게 된다.

  • Controller에서 수행한 로직이 완료되면 DispatcherServlet에 리턴되고 리턴된 결과는 ViewResolver를 통해서 보여지게 된다.

HandlerMapping

  • 여기서 spring web을 사용하게 되면 webmvcConfigurationSupport에 정의한 default handlerMapping bean이 등록되게 된다.

    @Bean
    public BeanNameUrlHandlerMapping beanNameHandlerMapping(
    		@Qualifier("mvcConversionService") FormattingConversionService conversionService,
    		@Qualifier("mvcResourceUrlProvider") ResourceUrlProvider resourceUrlProvider) {
    
    	BeanNameUrlHandlerMapping mapping = new BeanNameUrlHandlerMapping();
    	mapping.setOrder(2);
    
    	...
    }
    
    @Bean
    @Primary
    @Override
    public RequestMappingHandlerMapping requestMappingHandlerMapping(
    		@Qualifier("mvcContentNegotiationManager") ContentNegotiationManager contentNegotiationManager,
    		@Qualifier("mvcConversionService") FormattingConversionService conversionService,
    		@Qualifier("mvcResourceUrlProvider") ResourceUrlProvider resourceUrlProvider) {
    	// Must be @Primary for MvcUriComponentsBuilder to work
    	return super.requestMappingHandlerMapping(contentNegotiationManager, conversionService,
    			resourceUrlProvider);
    }
  • 기본 RequestHandler 종류

    • BeanNameUrlHandlerMapping

      @Controller("/accounts")
      public class AccountController {
      
      }
    • SimpleUrlHandlerMapping

      @Bean
      SimpleUrlHandlerMapping urlHandlerMapping() {
          SimpleUrlHandlerMapping simpleUrlHandlerMapping = new SimpleUrlHandlerMapping();
          simpleUrlHandlerMapping.setOrder(Ordered.LOWEST_PRECEDENCE - 2);
      
          Map<String, Object> mapping = new HashMap<>();
          mapping.put("/accounts", accountController());
          simpleUrlHandlerMapping.setUrlMap(mapping);
      
      		//or 
      		//  Properties properties = new Properties();
      		//  properties.put("/accounts", "accountController");
      		//  simpleUrlHandlerMapping.setMappings(properties);
      
          return simpleUrlHandlerMapping;
      }
    • RequestMappingHandlerMapping

      @Controller
      @RequestMapping("/accounts")
      public class AccountController {
      	....
      }

RequestMappingHandlerAdaper

  • RequestMappingHandlerAdaper를 초기에 등록할 때 getDefaultArgumentResolvers()를 통해서 다양한 파라미터에 대한 ArgumentResolver를 등록하게 되는데 대략 30개정도 된다.

    rivate List<HandlerMethodArgumentResolver> getDefaultArgumentResolvers() {
    		List<HandlerMethodArgumentResolver> resolvers = new ArrayList<>(30);
    
    		// Annotation-based argument resolution
    		resolvers.add(new RequestParamMethodArgumentResolver(getBeanFactory(), false));
    		resolvers.add(new RequestParamMapMethodArgumentResolver());
    		resolvers.add(new PathVariableMethodArgumentResolver());
    		resolvers.add(new PathVariableMapMethodArgumentResolver());
    		resolvers.add(new MatrixVariableMethodArgumentResolver());
    		resolvers.add(new MatrixVariableMapMethodArgumentResolver());
    		resolvers.add(new ServletModelAttributeMethodProcessor(false));
    		resolvers.add(new RequestResponseBodyMethodProcessor(getMessageConverters(), this.requestResponseBodyAdvice));
    		resolvers.add(new RequestPartMethodArgumentResolver(getMessageConverters(), this.requestResponseBodyAdvice));
    		resolvers.add(new RequestHeaderMethodArgumentResolver(getBeanFactory()));
    		resolvers.add(new RequestHeaderMapMethodArgumentResolver());
    		resolvers.add(new ServletCookieValueMethodArgumentResolver(getBeanFactory()));
    		
    		...
    }
  • 그중에는 RequestResponseBodyMethodProcessor 빈도 등록되는것을 확인할 수 있다.

DispatcherServlet

  • 그럼 이제 HandlerMapping이 등록된다는 것을 알수 있었고, ArgumentResolver도 HandlerMappingAdaper에 종속되어 등록되는 것을 알 수 있다.

  • 그럼 이제 사용자가 호출하게 되면 어떤 과정을 거치게 될까?

  • 처음에 사용자 요청을 통해서 Filter를 지나 DispatcherServlet으로 오게 되면

  • doService → doDispatch()를 통해서 등록한 Handler중에 적합한 Handler를 찾게 된다.

    mappedHandler = getHandler(processedRequest);
  • doDispatch()안에 있는 getHandler() 메소드를 통해서 RequestmappingHandle이라는 빈이 있다는 것을 확인하게 되고 리턴한다.

    @Nullable
    	protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
    		if (this.handlerMappings != null) {
    			for (HandlerMapping mapping : this.handlerMappings) {
    				HandlerExecutionChain handler = mapping.getHandler(request);
    				if (handler != null) {
    					return handler;
    				}
    			}
    		}
    		return null;
    	}
  • 그럼 handle()을 호출하여 추상화된 AbstractHandlerMethodAdapter 객체는 Handler와 관련되어 있는 Adapter 객체를 호출하게 된다.

    // DispatcherServlet.doDispatch
    mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
    
    // AbstractHandlerMethodAdapter.handler
    public final ModelAndView handle(HttpServletRequest request, HttpServletResponse response, Object handler)
    			throws Exception {
    
    		return handleInternal(request, response, (HandlerMethod) handler);
    	}
  • 그렇게 확인된 RequestMappingHandlerAdaper 객체는 interceptor 과정을 거치고 나서 등록된 request 유형에 따라 argumentResolver를 호출하게 된다.

    // RequestMappingHandlerAdaper
    public Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer,
    			NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception {
    
    		HandlerMethodArgumentResolver resolver = getArgumentResolver(parameter);
    		if (resolver == null) {
    			throw new IllegalArgumentException("Unsupported parameter type [" +
    					parameter.getParameterType().getName() + "]. supportsParameter should be called first.");
    		}
    		return resolver.resolveArgument(parameter, mavContainer, webRequest, binderFactory);
    	}
  • RequestResponseBodyMethodProcessor.resolverArgument() 메소드를 호출하게 되는데

    public Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer,
    			NativeWebRequest request, @Nullable WebDataBinderFactory binderFactory) throws Exception {
    		...
    		validateIfApplicable(binder, parameter);
    }
  • 그 안에는 AbstreacMessageConverterMethodArgumentResolver.validateIfApplicable 메소드를 호출하는 영역이 있는데 해당 로직에 valid 어노테이션에 대한 로직이 있는 것을 확인할 수 있다.

@Valid, @Validated의 차이는 무엇인가?

  • @Valid는 JSR 303 방식의 빈 검증 표준 스펙을 검증하는 자바 기본 라이브러리에서 제공해주는 어노테이션이다.

  • @Validated는 javax 라이브러리가 아닌 springframework에서 제공하는 어노테이션으로 특정 검증로직만 수행하고 싶거나 컨트롤러가 아닌 다른 레이어에서 검증 로직을 수행하고 싶을 경우 사용하는 어노테이션이다.

  • 그리고 @Validated는 ArgumentResolver를 통해서 구현되는 로직이 아니라 AOP를 통해서 동작하기 때문에 클래스 내부 메소드 호출은 프록시 적용이 되지 않으므로 동작하지 않을수도 있으니 주의해야 한다.

    • 부분 검증 예제

      public class ValidationGroups {
          public interface group1 {};
          public interface group2 {};
      }
      
      public class User {
          private Integer id;
          @NotEmpty(groups = {ValidationGroups.group1.class})
          private String account;
          @NotEmpty(groups = {ValidationGroups.group2.class})
          private String password;
      }
      
      @RequestMapping(value = "/user/login", method = RequestMethod.POST)
      public ResponseEntity login(@RequestBody @Validated(ValidationGroups.group1.class) User user, BindingResult bindingResult) {
          // do something
      }
    • 서비스 레이어에서 사용하는 예제

      @Service
      @Validated
      public ItemService {
      	public Item saveItem(@Valid Item) {
      		...
      	}
      }

참고

https://mangkyu.tistory.com/174
https://joont92.github.io/spring/HandlerMapping-HandlerAdapter-HandlerInterceptor/
https://cprayer.github.io/posts/about-spring-request-mapping-in-servlet-context/
https://maenco.tistory.com/entry/Spring-MVC-Argument-Resolver%EC%99%80-ReturnValue-Handler?category=959609
https://supawer0728.github.io/2018/04/04/spring-filter-interceptor/
https://maenco.tistory.com/entry/Spring-MVC-Argument-Resolver%EC%99%80-ReturnValue-Handler?category=959609