ABOUT ME

-

Today
-
Yesterday
-
Total
-
  • [모각코 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 구조
      • 동작순서
        1. 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회한다.
        2. 핸들러 어댑터 조회 : 핸들러를 실행할 수 있는 어댑터를 조회한다.
        3. 핸들러 어댑터 실행 : 핸들러 어댑터를 실행한다.
        4. 핸들러 실행 : 핸들러 어댑터가 실제 핸들러를 실행한다.
        5. ModelAndView 반환 : 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.
        6. viewResolver 호출 : 뷰 리졸버를 찾고 실행한다.
        7. View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다.
        8. 뷰 렌더링 : 뷰를 통해서 뷰를 렌더링한다.
    • 인터페이스
      • 스프링 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 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행해야한다.
      • 실행 순서
        1. 핸들러 매핑으로 핸들러 조회
          1. HandlerMapping을 순서대로 실행해서 핸들러를 찾는다.
          2. Controller 인터페이스를 구현하는 경우 빈의 이름으로 핸들러를 찾아야 하기 때문에 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping을 실행하게되고, 핸들러인 OldController을 반환한다.
        2. 핸들러 어댑터 조회
          1. HandlerAdapter의 supports 메서드를 순서대로 호출한다.
          2. SimpleControllerHandlerAdapter가 Controller 인터페이스를 지원하므로 대상이 된다.
        3. 핸들러 어댑터 실행
          1. 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하면서 핸들러 정보를 넘겨준다.
          2. 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");
            }
      }
      
      • 실행 순서
        1. 핸들러 매핑으로 핸들러 조회
          1. 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping가 실행되고, MyHttpRequestHandler를 반환한다.
        2. 핸들러 어댑터 조회
          1. HandlerAdapter의 supports가 순서대로 호출된다.
          2. HttpRequestHandlerAdapter가 HttpRequestHandler 인터페이스를 지원하므로 대상이 된다.
        3. 핸들러 어댑터 실행
          1. 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하고 핸들러 정보를 넘겨준다.
          2. HttpRequestHandlerAdapter가 핸들러인 MyHttpRequestHandler을 실행하고 결과를 반환한다.

    ViewResolver

    • 스프링 부트가 자동 등록하는 뷰 리졸버
    • 동작 과정
      1. 핸들러 어댑터 호출
        1. 핸들러 어댑터를 통해 논리 이름인 “new-form”을 획득한다.
      2. ViewResolver 호출
        1. new-form이라는 뷰 이름으로 viewResolver를 순서대로 호출한다.
        2. BeanNameViewResolver는 new-form이라는 이름으로 스프링 빈으로 등록된 뷰를 찾아야한다.
        3. 없다면 InternalResourceViewResolver가 호출된다.
      3. InternalResourceViewResolver
        1. 이 뷰 리졸버는 InternalResourceView를 반환한다.
      4. InternalResourceView
        1. 해당 뷰는 JSP처럼 포워드를 호출해서 처리할 수 있는 경우에 사용한다
      5. view.render()
        1. 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 구조
      • 동작순서
        1. 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회한다.
        2. 핸들러 어댑터 조회 : 핸들러를 실행할 수 있는 어댑터를 조회한다.
        3. 핸들러 어댑터 실행 : 핸들러 어댑터를 실행한다.
        4. 핸들러 실행 : 핸들러 어댑터가 실제 핸들러를 실행한다.
        5. ModelAndView 반환 : 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.
        6. viewResolver 호출 : 뷰 리졸버를 찾고 실행한다.
        7. View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다.
        8. 뷰 렌더링 : 뷰를 통해서 뷰를 렌더링한다.
    • 인터페이스
      • 스프링 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 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행해야한다.
      • 실행 순서
        1. 핸들러 매핑으로 핸들러 조회
          1. HandlerMapping을 순서대로 실행해서 핸들러를 찾는다.
          2. Controller 인터페이스를 구현하는 경우 빈의 이름으로 핸들러를 찾아야 하기 때문에 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping을 실행하게되고, 핸들러인 OldController을 반환한다.
        2. 핸들러 어댑터 조회
          1. HandlerAdapter의 supports 메서드를 순서대로 호출한다.
          2. SimpleControllerHandlerAdapter가 Controller 인터페이스를 지원하므로 대상이 된다.
        3. 핸들러 어댑터 실행
          1. 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하면서 핸들러 정보를 넘겨준다.
          2. 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");
            }
      }
      
      • 실행 순서
        1. 핸들러 매핑으로 핸들러 조회
          1. 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping가 실행되고, MyHttpRequestHandler를 반환한다.
        2. 핸들러 어댑터 조회
          1. HandlerAdapter의 supports가 순서대로 호출된다.
          2. HttpRequestHandlerAdapter가 HttpRequestHandler 인터페이스를 지원하므로 대상이 된다.
        3. 핸들러 어댑터 실행
          1. 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하고 핸들러 정보를 넘겨준다.
          2. HttpRequestHandlerAdapter가 핸들러인 MyHttpRequestHandler을 실행하고 결과를 반환한다.

    ViewResolver

    • 스프링 부트가 자동 등록하는 뷰 리졸버
    • 동작 과정
      1. 핸들러 어댑터 호출
        1. 핸들러 어댑터를 통해 논리 이름인 “new-form”을 획득한다.
      2. ViewResolver 호출
        1. new-form이라는 뷰 이름으로 viewResolver를 순서대로 호출한다.
        2. BeanNameViewResolver는 new-form이라는 이름으로 스프링 빈으로 등록된 뷰를 찾아야한다.
        3. 없다면 InternalResourceViewResolver가 호출된다.
      3. InternalResourceViewResolver
        1. 이 뷰 리졸버는 InternalResourceView를 반환한다.
      4. InternalResourceView
        1. 해당 뷰는 JSP처럼 포워드를 호출해서 처리할 수 있는 경우에 사용한다
      5. view.render()
        1. 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 구조
      • 동작순서
        1. 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회한다.
        2. 핸들러 어댑터 조회 : 핸들러를 실행할 수 있는 어댑터를 조회한다.
        3. 핸들러 어댑터 실행 : 핸들러 어댑터를 실행한다.
        4. 핸들러 실행 : 핸들러 어댑터가 실제 핸들러를 실행한다.
        5. ModelAndView 반환 : 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.
        6. viewResolver 호출 : 뷰 리졸버를 찾고 실행한다.
        7. View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다.
        8. 뷰 렌더링 : 뷰를 통해서 뷰를 렌더링한다.
    • 인터페이스
      • 스프링 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 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행해야한다.
      • 실행 순서
        1. 핸들러 매핑으로 핸들러 조회
          1. HandlerMapping을 순서대로 실행해서 핸들러를 찾는다.
          2. Controller 인터페이스를 구현하는 경우 빈의 이름으로 핸들러를 찾아야 하기 때문에 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping을 실행하게되고, 핸들러인 OldController을 반환한다.
        2. 핸들러 어댑터 조회
          1. HandlerAdapter의 supports 메서드를 순서대로 호출한다.
          2. SimpleControllerHandlerAdapter가 Controller 인터페이스를 지원하므로 대상이 된다.
        3. 핸들러 어댑터 실행
          1. 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하면서 핸들러 정보를 넘겨준다.
          2. 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");
            }
      }
      
      • 실행 순서
        1. 핸들러 매핑으로 핸들러 조회
          1. 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping가 실행되고, MyHttpRequestHandler를 반환한다.
        2. 핸들러 어댑터 조회
          1. HandlerAdapter의 supports가 순서대로 호출된다.
          2. HttpRequestHandlerAdapter가 HttpRequestHandler 인터페이스를 지원하므로 대상이 된다.
        3. 핸들러 어댑터 실행
          1. 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하고 핸들러 정보를 넘겨준다.
          2. HttpRequestHandlerAdapter가 핸들러인 MyHttpRequestHandler을 실행하고 결과를 반환한다.

    ViewResolver

    • 스프링 부트가 자동 등록하는 뷰 리졸버
    • 동작 과정
      1. 핸들러 어댑터 호출
        1. 핸들러 어댑터를 통해 논리 이름인 “new-form”을 획득한다.
      2. ViewResolver 호출
        1. new-form이라는 뷰 이름으로 viewResolver를 순서대로 호출한다.
        2. BeanNameViewResolver는 new-form이라는 이름으로 스프링 빈으로 등록된 뷰를 찾아야한다.
        3. 없다면 InternalResourceViewResolver가 호출된다.
      3. InternalResourceViewResolver
        1. 이 뷰 리졸버는 InternalResourceView를 반환한다.
      4. InternalResourceView
        1. 해당 뷰는 JSP처럼 포워드를 호출해서 처리할 수 있는 경우에 사용한다
      5. view.render()
        1. 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 구조
      • 동작순서
        1. 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회한다.
        2. 핸들러 어댑터 조회 : 핸들러를 실행할 수 있는 어댑터를 조회한다.
        3. 핸들러 어댑터 실행 : 핸들러 어댑터를 실행한다.
        4. 핸들러 실행 : 핸들러 어댑터가 실제 핸들러를 실행한다.
        5. ModelAndView 반환 : 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView로 변환해서 반환한다.
        6. viewResolver 호출 : 뷰 리졸버를 찾고 실행한다.
        7. View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다.
        8. 뷰 렌더링 : 뷰를 통해서 뷰를 렌더링한다.
    • 인터페이스
      • 스프링 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 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행해야한다.
      • 실행 순서
        1. 핸들러 매핑으로 핸들러 조회
          1. HandlerMapping을 순서대로 실행해서 핸들러를 찾는다.
          2. Controller 인터페이스를 구현하는 경우 빈의 이름으로 핸들러를 찾아야 하기 때문에 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping을 실행하게되고, 핸들러인 OldController을 반환한다.
        2. 핸들러 어댑터 조회
          1. HandlerAdapter의 supports 메서드를 순서대로 호출한다.
          2. SimpleControllerHandlerAdapter가 Controller 인터페이스를 지원하므로 대상이 된다.
        3. 핸들러 어댑터 실행
          1. 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하면서 핸들러 정보를 넘겨준다.
          2. 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");
            }
      }
      
      • 실행 순서
        1. 핸들러 매핑으로 핸들러 조회
          1. 빈 이름으로 핸들러를 찾아주는 BeanNameUrlHandlerMapping가 실행되고, MyHttpRequestHandler를 반환한다.
        2. 핸들러 어댑터 조회
          1. HandlerAdapter의 supports가 순서대로 호출된다.
          2. HttpRequestHandlerAdapter가 HttpRequestHandler 인터페이스를 지원하므로 대상이 된다.
        3. 핸들러 어댑터 실행
          1. 디스패처 서블릿이 조회한 핸들러 어댑터를 실행하고 핸들러 정보를 넘겨준다.
          2. HttpRequestHandlerAdapter가 핸들러인 MyHttpRequestHandler을 실행하고 결과를 반환한다.

    ViewResolver

    • 스프링 부트가 자동 등록하는 뷰 리졸버
    • 동작 과정
      1. 핸들러 어댑터 호출
        1. 핸들러 어댑터를 통해 논리 이름인 “new-form”을 획득한다.
      2. ViewResolver 호출
        1. new-form이라는 뷰 이름으로 viewResolver를 순서대로 호출한다.
        2. BeanNameViewResolver는 new-form이라는 이름으로 스프링 빈으로 등록된 뷰를 찾아야한다.
        3. 없다면 InternalResourceViewResolver가 호출된다.
      3. InternalResourceViewResolver
        1. 이 뷰 리졸버는 InternalResourceView를 반환한다.
      4. InternalResourceView
        1. 해당 뷰는 JSP처럼 포워드를 호출해서 처리할 수 있는 경우에 사용한다
      5. view.render()
        1. render 메서드가 호출되고, InternalResourceView는 forward를 호출해서 JSP를 실행한다.
Designed by Tistory.