본문 바로가기

프로그래밍/Spring

[SPRING] MVC 구조 이해

스프링 MVC 전체 구조

DispatcherServlet, HandlerMapping, HandlerAdapter, ModelAndView, ViewResolver, View

 

고객의 요청이 오면 DispatcherServlet이 받아서 스프링부트가 미리 등록해둔 여러개의 핸들러를 HandlerMapping이 처리할 수 있는 핸들러를 찾아주고 걔를 HandlerAdapter목록에 던져서 처리할 수 있는 HandlerAdapter찾아 호출(handle), 반환(ModelAndView)를 하고 viewResolver를 호출해서 실제 view를 찾아서 render 호출

 

- DispatcherServlet

org.springframework.web.servlet.DispatcherServlet

 

1. DispatcherServlet 스프링 MVC의 핵심이며, DispatcherServlet도 HttpServlet을 상속 받아서 사용하고 Servlet이다.

2. 스프링 부트가 내장 톰켓을 띄우면서 DispatcherServlet을 서블릿으로 자동으로 등록하면서 모든 경로(urlPatterns="/")에 대해서 매핑한다.

3. 더 자세한 경로가 우선순위가 높다.

 

요청흐름

1. 서블릿이 호출되면 HttpServlet 이 제공하는 serivce() 가 호출된다.

2. 스프링 MVC는 DispatcherServlet 의 부모인 FrameworkServlet 에서 service() 를 오버라이드 해두었다.

3. FrameworkServlet.service() 를 시작으로 여러 메서드가 호출되면서 DispacherServlet.doDispatch() 가 호출된다.

 

동작 순서

1. 핸들러 조회: 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러(컨트롤러)를 조회한다.

2. 핸들러 어댑터 조회: 핸들러를 실행할 수 있는 핸들러 어댑터를 조회한다.

3. 핸들러 어댑터 실행: 핸들러 어댑터를 실행한다.

4. 핸들러 실행: 핸들러 어댑터가 실제 핸들러를 실행한다.

5. ModelAndView 반환: 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.

6. viewResolver 호출: 뷰 리졸버를 찾고 실행한다. JSP의 경우: InternalResourceViewResolver 가 자동 등록되고, 사용된다.

7. View 반환: 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다. JSP의 경우 InternalResourceView(JstlView) 를 반환하는데, 내부에 forward() 로직이 있다.

8. 뷰 렌더링: 뷰를 통해서 뷰를 렌더링 한다.

 

인터페이스 살펴보기

스프링 MVC의 큰 강점은 DispatcherServlet 코드의 변경 없이, 원하는 기능을 변경하거나 확장할 수 있다는 점이다. 지금까지 설명한 대부분을 확장 가능할 수 있게 인터페이스로 제공한다.

이 인터페이스들만 구현해서 DispatcherServlet 에 등록하면 여러분만의 컨트롤러를 만들 수도 있다.

 

주요 인터페이스 목록

핸들러 매핑: org.springframework.web.servlet.HandlerMapping

핸들러 어댑터: org.springframework.web.servlet.HandlerAdapter

뷰 리졸버: org.springframework.web.servlet.ViewResolver

뷰: org.springframework.web.servlet.View

 

 

핸들러 매핑과 핸들러 어댑터

스프링은 과거에 Controller라는 인터페이스가 있어서 애노테이션이 아닌 Controller 인터페이스 사용

org.springframework.web.servlet.mvc.Controller 

public interface Controller { 
	ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception; 
}

 

HandlerMapping

0 = RequestMappingHandlerMapping : 애노테이션 기반의 컨트롤러인 @RequestMapping에서 사용

1 = BeanNameUrlHandlerMapping : 스프링 빈의 이름으로 핸들러를 찾는다.

...

 

HandlerAdapter

0 = RequestMappingHandlerAdapter : 애노테이션 기반의 컨트롤러인 @RequestMapping에서 사용

1 = HttpRequestHandlerAdapter : HttpRequestHandler 처리

2 = SimpleControllerHandlerAdapter : Controller 인터페이스(애노테이션X, 과거에 사용) 처리

...

 

뷰 리졸버

스프링 부트는 InternalResourceViewResolver 라는 뷰 리졸버를 자동으로 등록하는데, 이때 application.properties 에 등록한 spring.mvc.view.prefix , spring.mvc.view.suffix 설정 정보를 사용해서 등록한다.

 

스프링 부트가 자동 등록하는 뷰 리졸버

1 = BeanNameViewResolver : 빈 이름으로 뷰를 찾아서 반환한다. (예: 엑셀 파일 생성 기능에 사용)

2 = InternalResourceViewResolver : JSP를 처리할 수 있는 뷰를 반환한다.

...