빙응의 공부 블로그

[Spring]스프링 MVC 1편 - 구조 이해 본문

Spring/인프런_개념

[Spring]스프링 MVC 1편 - 구조 이해

빙응이 2024. 1. 11. 16:47

📝MVC 구조의 이해 

우리가 만들었던 구조와 실제 스프링 MVC의 구조를 비교해보자! 

각각의 기능이 매핑된다.

  • FrontController - DispatcherServlet
  • handlerMappingMap - HandlerMapping
  • ModelView - ModelAndView
  • viewResolver(메소드) - ViewResolver(인터페이스)
  • MyView(객체)-View(인터페이스)
DispatcherServlet

스프링 MVC도 프론트 컨트롤러 패턴으로 구현되어 있다.

스프링 MVC의 프론트 컨트롤러가 바로 DispatherServlert이다.

  • DispatcherServlet도 부모 클래스에서 HttpServlet을 상속 받아서 사용하며, 서블릿으로 동작한다.
  • 스프링 부트는 DispatcherServlet을 서블릿으로 자동을 등록하고 모든 경로(urlPatterns="/")에 매핑한다.

요청 흐름

  • 서블릿이 호출되면 HttpServlet이 제공하는 service()가 호출된다
  • 스프링 MVC는 DispatcherServlet의 부모인 FrameworkServlet에서 service()를 오버라이딩해서 사용한다.
  • FrameworkeServlet.service()를 시작으로 여러 메서드가 호출되면서 DispatcherServlet.doDispatch()를 실행한다.

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);
}

 

MVC 동작 순서

  1. 핸들러 조회 : 핸들러 매핑을 통해 요청 URL에 매핑된 핸들러를 조회
  2. 핸들러 어댑터 조회
  3. 핸들러 어댑터 실행
  4. 핸들러 실행
  5. ModelAndView 반환
  6. ViewResolver 호출
  7. View 반환 : 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 랜더링 역할을 하는 뷰 객체를 반환한다. 
  8. 뷰 렌더링

인터페이스 살펴보기 

  • 스프링 MVC의 큰 장점은 DispatcherServlet 코드의 변경 없이, 원하는 기능을 변경하거나 확장 가능하다.
정리

스프링 MVC는 코드 분량이 많고, 내부가 복잡하다. 사실 해당 기능을 확장하거나 나만의 컨트롤러를 만들 일은 없다고 한다.... 핵심 동작방식을 파악하여 문제를 쉽게 해결하도록 하자~ 

 

📝핸들러 매핑과 핸들러 어댑터 - Controller 인터페이스 

핸들러 매핑과 핸들러 어댑터가 어떤 것들이 있는지 알아보자,

현재는 사용안한다! 

과거 Controller 인터페이스
public interface Controller {
     ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception;
}


//실습코드
@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;
  }
}
  • 해당 코드에서 @Component로 /springmvc/old-controller라는 이름의 스프링 빈을 등록하였다.
  • 빈의 이름으로 URL 매핑을 할 수 있다.!
이 컨트롤러는 어떻게 호출될 수 있었을까?

이 컨트롤러가 호출되려면 다음 2가지가 필요하다.

  • HandlerMapping(핸들러 매핑)
    • 핸들러 매핑에서 이 컨트롤러를 찾을 수 있어야 한다.
    • 예)스프링 빈의 이름으로 핸들러를 찾을 수 있는 핸들러 매핑이 필요 
  • HandlerAdapter(핸들러 어댑터)
    • 핸들러 매핑을 통해서 찾은 핸들러를 실행할 수 있는 핸들러 어댑터가 필요하다.
    • 예) Controller 인터페이스를 실행할 수 있는 핸들러 어댑터를 찾고 실행한다. 
스프링 MVC의 기능으로 2가지 구현 

HandlerMapping

0 = RequestMappingHandlerMapping     : 애노테이션 기반의 컨트롤러인 @RequestMapping에서 사용한다.
1 = BeanNameUrlHandlerMapping          : 스프링 빈의 이름으로 핸들러를 찾는다.   

 

HandlerAdapter

0 = RequestMappingHandlerAdapter : 애노테이션 기반의 컨트롤러인 @RequestMapping에서 사용
1 = HttpRequestHandlerAdapter : HttpRequestHandler 처리
2 = SimpleControllerHandlerAdapter : Controller 인터페이스(애노테이션X, 과거에 사용) 처리

 

정리 - OldController 핸들러 매핑, 어댑터
  • HandlerMapping = BeanNameUrlHandlerMapping 사용 
  • HandlerAdapter = SimpleControllerHandlerAdapter 사용 

 

📝핸들러 매핑과 핸들러 어댑터 - HttpRequestHandler 

HttpRequestHandler 핸들러는 서블릿과 가장 유사한 형태의 핸들러이다. 

public interface HttpRequestHandler {
      void handleRequest(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException;
}

//실습코드
@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");
  }
}

 

정리 - HttpRequestHandler 핸들러 매핑, 어댑터
  • HandlerMapping = BeanNameUrlHandlerMapping 사용
  • HandlerAdapter = HttpRequestHandlerAdapter 사용 

📝뷰 리졸버

@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 new ModelAndView("new-form");
 }
}

해당 코드를 실행해보면 정상작동은 하지만 View를 찾지 못해  Whitelabel Error Page 오류가 발생한다.

뷰 리졸버 - InternalResourceViewResolver

스프링 부트는 뷰 리졸버를 자동으로 등록하며 밑의 코드를 통해 설정 정보를 저장한다. 

spring.mvc.view.prefix=/WEB-INF/views/
spring.mvc.view.suffix=.jsp

 

동작 방식
  • 스프링 부트가 자동 등록하는 뷰 리졸버
1 = BeanNameViewResolver : 빈 이름으로 뷰를 찾아서 반환한다. (예: 엑셀 파일 생성 기능 에 사용)
2 = InternalResourceViewResolver : JSP를 처리할 수 있는 뷰를 반환한다