-
[모각코 10주차] 결과 (16:00 ~ 19:00)카테고리 없음 2022. 9. 9. 14:57
Spring MVC
- 구조
- DispatcherServlet
- org.springframework.web.servlet.DispatcherServlet
- 스프링 MVC의 프론트 컨트롤러가 디스패처 서블릿이다.
- DispatcherServlet도 부모 클래스에서 HttpServlet을 상속받아서 사용하고, 서블릿으로 동작한다
- DispatcherServlet → FrameworkServlet → HttpServletBean → HttpServlet
- 스프링 부트는 DispatcherServlet을 서블릿으로 자동으로 등록하면서 모든 경로 (urlPattern = “/”)에 대해서 매핑한다.
- 요청 흐름
- 서블릿이 호출되면 HttpServlet이 제공하는 service()가 호출된다.
- 스프링 MVC는 DispatcherServlet의 부모인 FrameworkServlet에서 service()를 오버라이드 해두었다.
- FrameworkServlet.service()를 시작으로 여러 메서드가 호출되면서 DispatcherServlet.doDispatch가 호출된다.
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception { HttpServletRequest processedRequest = request; HandlerExecutionChain mappedHandler = null; ModelAndView mv = null; // 1. 핸들러 조회 mappedHandler = getHandler(processedRequest); if (mappedHandler == null) { noHandlerFound(processedRequest, response); return; } //2.핸들러 어댑터 조회-핸들러를 처리할 수 있는 어댑터 HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler()); // 3. 핸들러 어댑터 실행 -> 4. 핸들러 어댑터를 통해 핸들러 실행 -> 5. ModelAndView 반환 mv = ha.handle(processedRequest, response, mappedHandler.getHandler()); processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException); } private void processDispatchResult(HttpServletRequest request, HttpServletResponse response, HandlerExecutionChain mappedHandler, ModelAndView mv, Exception exception) throws Exception { // 뷰 렌더링 호출 render(mv, request, response); } protected void render(ModelAndView mv, HttpServletRequest request, HttpServletResponse response) throws Exception { View view; String viewName = mv.getViewName(); //6. 뷰 리졸버를 통해서 뷰 찾기,7.View 반환 view = resolveViewName(viewName, mv.getModelInternal(), locale, request); // 8. 뷰 렌더링 view.render(mv.getModelInternal(), request, response); }
- SpringMVC 구조
- 동작순서
- 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회한다.
- 핸들러 어댑터 조회 : 핸들러를 실행할 수 있는 어댑터를 조회한다.
- 핸들러 어댑터 실행 : 핸들러 어댑터를 실행한다.
- 핸들러 실행 : 핸들러 어댑터가 실제 핸들러를 실행한다.
- ModelAndView 반환 : 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.
- viewResolver 호출 : 뷰 리졸버를 찾고 실행한다.
- View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다.
- 뷰 렌더링 : 뷰를 통해서 뷰를 렌더링한다.
- 동작순서
- 인터페이스
- 스프링 MVC는 DispatcherServlet 코드의 변경 없이, 원하는 기능을 변경하거나 확장할 수 있도록 인터페이스를 제공한다.
- 핸들러 매핑, 핸들러 어댑터, 뷰 리졸버, 뷰가 인터페이스를 상속받아 구현되도록 되어있다.
핸들러 매핑과 핸들러 어댑터
- 스프링 부트가 자동으로 등록하는 핸들러 매핑과 핸들러 어댑터
- 핸들러 매핑
- 핸들러 어댑터
- Controller 인터페이스 (과거)
- org.springframework.web.servlet.mvc.Controller 인터페이스를 구현하여 컨트롤러를 만드는 방식
- Controller 인터페이스는 @Controller 애노테이션과는 전혀 다르다
@Component("/springmvc/old-controller") public class OldController implements Controller { @Override public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception { System.out.println("OldController.handleRequest"); return null; } }
- 이 컨트롤러는 /springmvc/old-controller 이라는 이름의 스프링 빈으로 등록되어있다.
- 빈의 이름으로 URL을 매핑한다.
- HandlerMapping
- 스프링 빈의 이름으로 핸들러를 찾을 수 있는 핸들러 매핑이 필요하다.
- HandlerAdapter
- 핸들러 매핑을 통해서 찾은 핸들러를 실행할 수 있는 어댑터가 필요하다
- Controller 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행해야한다.
- 실행 순서
- 핸들러 매핑으로 핸들러 조회
- HandlerMapping을 순서대로 실행해서 핸들러를 찾는다.
- Controller 인터페이스를 구현하는 경우 빈의 이름으로 핸들러를 찾아야 하기 때문에 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping을 실행하게되고, 핸들러인 OldController을 반환한다.
- 핸들러 어댑터 조회
- HandlerAdapter의 supports 메서드를 순서대로 호출한다.
- SimpleControllerHandlerAdapter가 Controller 인터페이스를 지원하므로 대상이 된다.
- 핸들러 어댑터 실행
- 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하면서 핸들러 정보를 넘겨준다.
- SimpleControllerHandlerAdapter가 핸들러인 OldController를 내부에서 실행하고 결과를 반환한다.
- 핸들러 매핑으로 핸들러 조회
- HttpRequestHandler (과거)
- HttpRequestHandler는 서블릿과 가장 유사한 형태의 핸들러이다.
@Component("/springmvc/request-handler") public class MyHttpRequestHandler implements HttpRequestHandler { @Override public void handleRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("MyHttpRequestHandler.handleRequest"); } }
- 실행 순서
- 핸들러 매핑으로 핸들러 조회
- 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping가 실행되고, MyHttpRequestHandler를 반환한다.
- 핸들러 어댑터 조회
- HandlerAdapter의 supports가 순서대로 호출된다.
- HttpRequestHandlerAdapter가 HttpRequestHandler 인터페이스를 지원하므로 대상이 된다.
- 핸들러 어댑터 실행
- 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하고 핸들러 정보를 넘겨준다.
- HttpRequestHandlerAdapter가 핸들러인 MyHttpRequestHandler을 실행하고 결과를 반환한다.
- 핸들러 매핑으로 핸들러 조회
ViewResolver
- 스프링 부트가 자동 등록하는 뷰 리졸버
- 동작 과정
- 핸들러 어댑터 호출
- 핸들러 어댑터를 통해 논리 이름인 “new-form”을 획득한다.
- ViewResolver 호출
- new-form이라는 뷰 이름으로 viewResolver를 순서대로 호출한다.
- BeanNameViewResolver는 new-form이라는 이름으로 스프링 빈으로 등록된 뷰를 찾아야한다.
- 없다면 InternalResourceViewResolver가 호출된다.
- InternalResourceViewResolver
- 이 뷰 리졸버는 InternalResourceView를 반환한다.
- InternalResourceView
- 해당 뷰는 JSP처럼 포워드를 호출해서 처리할 수 있는 경우에 사용한다
- view.render()
- render 메서드가 호출되고, InternalResourceView는 forward를 호출해서 JSP를 실행한다.
- 핸들러 어댑터 호출
Spring MVC
- 구조
- DispatcherServlet
- org.springframework.web.servlet.DispatcherServlet
- 스프링 MVC의 프론트 컨트롤러가 디스패처 서블릿이다.
- DispatcherServlet도 부모 클래스에서 HttpServlet을 상속받아서 사용하고, 서블릿으로 동작한다
- DispatcherServlet → FrameworkServlet → HttpServletBean → HttpServlet
- 스프링 부트는 DispatcherServlet을 서블릿으로 자동으로 등록하면서 모든 경로 (urlPattern = “/”)에 대해서 매핑한다.
- 요청 흐름
- 서블릿이 호출되면 HttpServlet이 제공하는 service()가 호출된다.
- 스프링 MVC는 DispatcherServlet의 부모인 FrameworkServlet에서 service()를 오버라이드 해두었다.
- FrameworkServlet.service()를 시작으로 여러 메서드가 호출되면서 DispatcherServlet.doDispatch가 호출된다.
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception { HttpServletRequest processedRequest = request; HandlerExecutionChain mappedHandler = null; ModelAndView mv = null; // 1. 핸들러 조회 mappedHandler = getHandler(processedRequest); if (mappedHandler == null) { noHandlerFound(processedRequest, response); return; } //2.핸들러 어댑터 조회-핸들러를 처리할 수 있는 어댑터 HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler()); // 3. 핸들러 어댑터 실행 -> 4. 핸들러 어댑터를 통해 핸들러 실행 -> 5. ModelAndView 반환 mv = ha.handle(processedRequest, response, mappedHandler.getHandler()); processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException); } private void processDispatchResult(HttpServletRequest request, HttpServletResponse response, HandlerExecutionChain mappedHandler, ModelAndView mv, Exception exception) throws Exception { // 뷰 렌더링 호출 render(mv, request, response); } protected void render(ModelAndView mv, HttpServletRequest request, HttpServletResponse response) throws Exception { View view; String viewName = mv.getViewName(); //6. 뷰 리졸버를 통해서 뷰 찾기,7.View 반환 view = resolveViewName(viewName, mv.getModelInternal(), locale, request); // 8. 뷰 렌더링 view.render(mv.getModelInternal(), request, response); }
- SpringMVC 구조
- 동작순서
- 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회한다.
- 핸들러 어댑터 조회 : 핸들러를 실행할 수 있는 어댑터를 조회한다.
- 핸들러 어댑터 실행 : 핸들러 어댑터를 실행한다.
- 핸들러 실행 : 핸들러 어댑터가 실제 핸들러를 실행한다.
- ModelAndView 반환 : 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.
- viewResolver 호출 : 뷰 리졸버를 찾고 실행한다.
- View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다.
- 뷰 렌더링 : 뷰를 통해서 뷰를 렌더링한다.
- 동작순서
- 인터페이스
- 스프링 MVC는 DispatcherServlet 코드의 변경 없이, 원하는 기능을 변경하거나 확장할 수 있도록 인터페이스를 제공한다.
- 핸들러 매핑, 핸들러 어댑터, 뷰 리졸버, 뷰가 인터페이스를 상속받아 구현되도록 되어있다.
핸들러 매핑과 핸들러 어댑터
- 스프링 부트가 자동으로 등록하는 핸들러 매핑과 핸들러 어댑터
- 핸들러 매핑
- 핸들러 어댑터
- Controller 인터페이스 (과거)
- org.springframework.web.servlet.mvc.Controller 인터페이스를 구현하여 컨트롤러를 만드는 방식
- Controller 인터페이스는 @Controller 애노테이션과는 전혀 다르다
@Component("/springmvc/old-controller") public class OldController implements Controller { @Override public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception { System.out.println("OldController.handleRequest"); return null; } }
- 이 컨트롤러는 /springmvc/old-controller 이라는 이름의 스프링 빈으로 등록되어있다.
- 빈의 이름으로 URL을 매핑한다.
- HandlerMapping
- 스프링 빈의 이름으로 핸들러를 찾을 수 있는 핸들러 매핑이 필요하다.
- HandlerAdapter
- 핸들러 매핑을 통해서 찾은 핸들러를 실행할 수 있는 어댑터가 필요하다
- Controller 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행해야한다.
- 실행 순서
- 핸들러 매핑으로 핸들러 조회
- HandlerMapping을 순서대로 실행해서 핸들러를 찾는다.
- Controller 인터페이스를 구현하는 경우 빈의 이름으로 핸들러를 찾아야 하기 때문에 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping을 실행하게되고, 핸들러인 OldController을 반환한다.
- 핸들러 어댑터 조회
- HandlerAdapter의 supports 메서드를 순서대로 호출한다.
- SimpleControllerHandlerAdapter가 Controller 인터페이스를 지원하므로 대상이 된다.
- 핸들러 어댑터 실행
- 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하면서 핸들러 정보를 넘겨준다.
- SimpleControllerHandlerAdapter가 핸들러인 OldController를 내부에서 실행하고 결과를 반환한다.
- 핸들러 매핑으로 핸들러 조회
- HttpRequestHandler (과거)
- HttpRequestHandler는 서블릿과 가장 유사한 형태의 핸들러이다.
@Component("/springmvc/request-handler") public class MyHttpRequestHandler implements HttpRequestHandler { @Override public void handleRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("MyHttpRequestHandler.handleRequest"); } }
- 실행 순서
- 핸들러 매핑으로 핸들러 조회
- 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping가 실행되고, MyHttpRequestHandler를 반환한다.
- 핸들러 어댑터 조회
- HandlerAdapter의 supports가 순서대로 호출된다.
- HttpRequestHandlerAdapter가 HttpRequestHandler 인터페이스를 지원하므로 대상이 된다.
- 핸들러 어댑터 실행
- 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하고 핸들러 정보를 넘겨준다.
- HttpRequestHandlerAdapter가 핸들러인 MyHttpRequestHandler을 실행하고 결과를 반환한다.
- 핸들러 매핑으로 핸들러 조회
ViewResolver
- 스프링 부트가 자동 등록하는 뷰 리졸버
- 동작 과정
- 핸들러 어댑터 호출
- 핸들러 어댑터를 통해 논리 이름인 “new-form”을 획득한다.
- ViewResolver 호출
- new-form이라는 뷰 이름으로 viewResolver를 순서대로 호출한다.
- BeanNameViewResolver는 new-form이라는 이름으로 스프링 빈으로 등록된 뷰를 찾아야한다.
- 없다면 InternalResourceViewResolver가 호출된다.
- InternalResourceViewResolver
- 이 뷰 리졸버는 InternalResourceView를 반환한다.
- InternalResourceView
- 해당 뷰는 JSP처럼 포워드를 호출해서 처리할 수 있는 경우에 사용한다
- view.render()
- render 메서드가 호출되고, InternalResourceView는 forward를 호출해서 JSP를 실행한다.
- 핸들러 어댑터 호출
Spring MVC
- 구조
- DispatcherServlet
- org.springframework.web.servlet.DispatcherServlet
- 스프링 MVC의 프론트 컨트롤러가 디스패처 서블릿이다.
- DispatcherServlet도 부모 클래스에서 HttpServlet을 상속받아서 사용하고, 서블릿으로 동작한다
- DispatcherServlet → FrameworkServlet → HttpServletBean → HttpServlet
- 스프링 부트는 DispatcherServlet을 서블릿으로 자동으로 등록하면서 모든 경로 (urlPattern = “/”)에 대해서 매핑한다.
- 요청 흐름
- 서블릿이 호출되면 HttpServlet이 제공하는 service()가 호출된다.
- 스프링 MVC는 DispatcherServlet의 부모인 FrameworkServlet에서 service()를 오버라이드 해두었다.
- FrameworkServlet.service()를 시작으로 여러 메서드가 호출되면서 DispatcherServlet.doDispatch가 호출된다.
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception { HttpServletRequest processedRequest = request; HandlerExecutionChain mappedHandler = null; ModelAndView mv = null; // 1. 핸들러 조회 mappedHandler = getHandler(processedRequest); if (mappedHandler == null) { noHandlerFound(processedRequest, response); return; } //2.핸들러 어댑터 조회-핸들러를 처리할 수 있는 어댑터 HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler()); // 3. 핸들러 어댑터 실행 -> 4. 핸들러 어댑터를 통해 핸들러 실행 -> 5. ModelAndView 반환 mv = ha.handle(processedRequest, response, mappedHandler.getHandler()); processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException); } private void processDispatchResult(HttpServletRequest request, HttpServletResponse response, HandlerExecutionChain mappedHandler, ModelAndView mv, Exception exception) throws Exception { // 뷰 렌더링 호출 render(mv, request, response); } protected void render(ModelAndView mv, HttpServletRequest request, HttpServletResponse response) throws Exception { View view; String viewName = mv.getViewName(); //6. 뷰 리졸버를 통해서 뷰 찾기,7.View 반환 view = resolveViewName(viewName, mv.getModelInternal(), locale, request); // 8. 뷰 렌더링 view.render(mv.getModelInternal(), request, response); }
- SpringMVC 구조
- 동작순서
- 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회한다.
- 핸들러 어댑터 조회 : 핸들러를 실행할 수 있는 어댑터를 조회한다.
- 핸들러 어댑터 실행 : 핸들러 어댑터를 실행한다.
- 핸들러 실행 : 핸들러 어댑터가 실제 핸들러를 실행한다.
- ModelAndView 반환 : 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.
- viewResolver 호출 : 뷰 리졸버를 찾고 실행한다.
- View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다.
- 뷰 렌더링 : 뷰를 통해서 뷰를 렌더링한다.
- 동작순서
- 인터페이스
- 스프링 MVC는 DispatcherServlet 코드의 변경 없이, 원하는 기능을 변경하거나 확장할 수 있도록 인터페이스를 제공한다.
- 핸들러 매핑, 핸들러 어댑터, 뷰 리졸버, 뷰가 인터페이스를 상속받아 구현되도록 되어있다.
핸들러 매핑과 핸들러 어댑터
- 스프링 부트가 자동으로 등록하는 핸들러 매핑과 핸들러 어댑터
- 핸들러 매핑
- 핸들러 어댑터
- Controller 인터페이스 (과거)
- org.springframework.web.servlet.mvc.Controller 인터페이스를 구현하여 컨트롤러를 만드는 방식
- Controller 인터페이스는 @Controller 애노테이션과는 전혀 다르다
@Component("/springmvc/old-controller") public class OldController implements Controller { @Override public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception { System.out.println("OldController.handleRequest"); return null; } }
- 이 컨트롤러는 /springmvc/old-controller 이라는 이름의 스프링 빈으로 등록되어있다.
- 빈의 이름으로 URL을 매핑한다.
- HandlerMapping
- 스프링 빈의 이름으로 핸들러를 찾을 수 있는 핸들러 매핑이 필요하다.
- HandlerAdapter
- 핸들러 매핑을 통해서 찾은 핸들러를 실행할 수 있는 어댑터가 필요하다
- Controller 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행해야한다.
- 실행 순서
- 핸들러 매핑으로 핸들러 조회
- HandlerMapping을 순서대로 실행해서 핸들러를 찾는다.
- Controller 인터페이스를 구현하는 경우 빈의 이름으로 핸들러를 찾아야 하기 때문에 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping을 실행하게되고, 핸들러인 OldController을 반환한다.
- 핸들러 어댑터 조회
- HandlerAdapter의 supports 메서드를 순서대로 호출한다.
- SimpleControllerHandlerAdapter가 Controller 인터페이스를 지원하므로 대상이 된다.
- 핸들러 어댑터 실행
- 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하면서 핸들러 정보를 넘겨준다.
- SimpleControllerHandlerAdapter가 핸들러인 OldController를 내부에서 실행하고 결과를 반환한다.
- 핸들러 매핑으로 핸들러 조회
- HttpRequestHandler (과거)
- HttpRequestHandler는 서블릿과 가장 유사한 형태의 핸들러이다.
@Component("/springmvc/request-handler") public class MyHttpRequestHandler implements HttpRequestHandler { @Override public void handleRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("MyHttpRequestHandler.handleRequest"); } }
- 실행 순서
- 핸들러 매핑으로 핸들러 조회
- 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping가 실행되고, MyHttpRequestHandler를 반환한다.
- 핸들러 어댑터 조회
- HandlerAdapter의 supports가 순서대로 호출된다.
- HttpRequestHandlerAdapter가 HttpRequestHandler 인터페이스를 지원하므로 대상이 된다.
- 핸들러 어댑터 실행
- 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하고 핸들러 정보를 넘겨준다.
- HttpRequestHandlerAdapter가 핸들러인 MyHttpRequestHandler을 실행하고 결과를 반환한다.
- 핸들러 매핑으로 핸들러 조회
ViewResolver
- 스프링 부트가 자동 등록하는 뷰 리졸버
- 동작 과정
- 핸들러 어댑터 호출
- 핸들러 어댑터를 통해 논리 이름인 “new-form”을 획득한다.
- ViewResolver 호출
- new-form이라는 뷰 이름으로 viewResolver를 순서대로 호출한다.
- BeanNameViewResolver는 new-form이라는 이름으로 스프링 빈으로 등록된 뷰를 찾아야한다.
- 없다면 InternalResourceViewResolver가 호출된다.
- InternalResourceViewResolver
- 이 뷰 리졸버는 InternalResourceView를 반환한다.
- InternalResourceView
- 해당 뷰는 JSP처럼 포워드를 호출해서 처리할 수 있는 경우에 사용한다
- view.render()
- render 메서드가 호출되고, InternalResourceView는 forward를 호출해서 JSP를 실행한다.
- 핸들러 어댑터 호출
Spring MVC
- 구조
- DispatcherServlet
- org.springframework.web.servlet.DispatcherServlet
- 스프링 MVC의 프론트 컨트롤러가 디스패처 서블릿이다.
- DispatcherServlet도 부모 클래스에서 HttpServlet을 상속받아서 사용하고, 서블릿으로 동작한다
- DispatcherServlet → FrameworkServlet → HttpServletBean → HttpServlet
- 스프링 부트는 DispatcherServlet을 서블릿으로 자동으로 등록하면서 모든 경로 (urlPattern = “/”)에 대해서 매핑한다.
- 요청 흐름
- 서블릿이 호출되면 HttpServlet이 제공하는 service()가 호출된다.
- 스프링 MVC는 DispatcherServlet의 부모인 FrameworkServlet에서 service()를 오버라이드 해두었다.
- FrameworkServlet.service()를 시작으로 여러 메서드가 호출되면서 DispatcherServlet.doDispatch가 호출된다.
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception { HttpServletRequest processedRequest = request; HandlerExecutionChain mappedHandler = null; ModelAndView mv = null; // 1. 핸들러 조회 mappedHandler = getHandler(processedRequest); if (mappedHandler == null) { noHandlerFound(processedRequest, response); return; } //2.핸들러 어댑터 조회-핸들러를 처리할 수 있는 어댑터 HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler()); // 3. 핸들러 어댑터 실행 -> 4. 핸들러 어댑터를 통해 핸들러 실행 -> 5. ModelAndView 반환 mv = ha.handle(processedRequest, response, mappedHandler.getHandler()); processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException); } private void processDispatchResult(HttpServletRequest request, HttpServletResponse response, HandlerExecutionChain mappedHandler, ModelAndView mv, Exception exception) throws Exception { // 뷰 렌더링 호출 render(mv, request, response); } protected void render(ModelAndView mv, HttpServletRequest request, HttpServletResponse response) throws Exception { View view; String viewName = mv.getViewName(); //6. 뷰 리졸버를 통해서 뷰 찾기,7.View 반환 view = resolveViewName(viewName, mv.getModelInternal(), locale, request); // 8. 뷰 렌더링 view.render(mv.getModelInternal(), request, response); }
- SpringMVC 구조
- 동작순서
- 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회한다.
- 핸들러 어댑터 조회 : 핸들러를 실행할 수 있는 어댑터를 조회한다.
- 핸들러 어댑터 실행 : 핸들러 어댑터를 실행한다.
- 핸들러 실행 : 핸들러 어댑터가 실제 핸들러를 실행한다.
- ModelAndView 반환 : 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.
- viewResolver 호출 : 뷰 리졸버를 찾고 실행한다.
- View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다.
- 뷰 렌더링 : 뷰를 통해서 뷰를 렌더링한다.
- 동작순서
- 인터페이스
- 스프링 MVC는 DispatcherServlet 코드의 변경 없이, 원하는 기능을 변경하거나 확장할 수 있도록 인터페이스를 제공한다.
- 핸들러 매핑, 핸들러 어댑터, 뷰 리졸버, 뷰가 인터페이스를 상속받아 구현되도록 되어있다.
핸들러 매핑과 핸들러 어댑터
- 스프링 부트가 자동으로 등록하는 핸들러 매핑과 핸들러 어댑터
- 핸들러 매핑
- 핸들러 어댑터
- Controller 인터페이스 (과거)
- org.springframework.web.servlet.mvc.Controller 인터페이스를 구현하여 컨트롤러를 만드는 방식
- Controller 인터페이스는 @Controller 애노테이션과는 전혀 다르다
@Component("/springmvc/old-controller") public class OldController implements Controller { @Override public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception { System.out.println("OldController.handleRequest"); return null; } }
- 이 컨트롤러는 /springmvc/old-controller 이라는 이름의 스프링 빈으로 등록되어있다.
- 빈의 이름으로 URL을 매핑한다.
- HandlerMapping
- 스프링 빈의 이름으로 핸들러를 찾을 수 있는 핸들러 매핑이 필요하다.
- HandlerAdapter
- 핸들러 매핑을 통해서 찾은 핸들러를 실행할 수 있는 어댑터가 필요하다
- Controller 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행해야한다.
- 실행 순서
- 핸들러 매핑으로 핸들러 조회
- HandlerMapping을 순서대로 실행해서 핸들러를 찾는다.
- Controller 인터페이스를 구현하는 경우 빈의 이름으로 핸들러를 찾아야 하기 때문에 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping을 실행하게되고, 핸들러인 OldController을 반환한다.
- 핸들러 어댑터 조회
- HandlerAdapter의 supports 메서드를 순서대로 호출한다.
- SimpleControllerHandlerAdapter가 Controller 인터페이스를 지원하므로 대상이 된다.
- 핸들러 어댑터 실행
- 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하면서 핸들러 정보를 넘겨준다.
- SimpleControllerHandlerAdapter가 핸들러인 OldController를 내부에서 실행하고 결과를 반환한다.
- 핸들러 매핑으로 핸들러 조회
- HttpRequestHandler (과거)
- HttpRequestHandler는 서블릿과 가장 유사한 형태의 핸들러이다.
@Component("/springmvc/request-handler") public class MyHttpRequestHandler implements HttpRequestHandler { @Override public void handleRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("MyHttpRequestHandler.handleRequest"); } }
- 실행 순서
- 핸들러 매핑으로 핸들러 조회
- 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping가 실행되고, MyHttpRequestHandler를 반환한다.
- 핸들러 어댑터 조회
- HandlerAdapter의 supports가 순서대로 호출된다.
- HttpRequestHandlerAdapter가 HttpRequestHandler 인터페이스를 지원하므로 대상이 된다.
- 핸들러 어댑터 실행
- 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하고 핸들러 정보를 넘겨준다.
- HttpRequestHandlerAdapter가 핸들러인 MyHttpRequestHandler을 실행하고 결과를 반환한다.
- 핸들러 매핑으로 핸들러 조회
ViewResolver
- 스프링 부트가 자동 등록하는 뷰 리졸버
- 동작 과정
- 핸들러 어댑터 호출
- 핸들러 어댑터를 통해 논리 이름인 “new-form”을 획득한다.
- ViewResolver 호출
- new-form이라는 뷰 이름으로 viewResolver를 순서대로 호출한다.
- BeanNameViewResolver는 new-form이라는 이름으로 스프링 빈으로 등록된 뷰를 찾아야한다.
- 없다면 InternalResourceViewResolver가 호출된다.
- InternalResourceViewResolver
- 이 뷰 리졸버는 InternalResourceView를 반환한다.
- InternalResourceView
- 해당 뷰는 JSP처럼 포워드를 호출해서 처리할 수 있는 경우에 사용한다
- view.render()
- render 메서드가 호출되고, InternalResourceView는 forward를 호출해서 JSP를 실행한다.
- 핸들러 어댑터 호출