[Spring] MVC 패턴 구현해보기

Manx·2022년 4월 29일
0

spring

목록 보기
12/24


Spring MVC 패턴을 직접 구현해보는 내용이 있어 듣고 정리해보았다.
강의에선 V1 ~ V5까지 다 구현을 하였는데, 내용이 너무 많아 V5의 내용만 포스팅할 예정이다.
V5의 핵심은 어느 종류의 컨트롤러도 유연하게 다룰 수 있다는 점이다. (끝판왕)

  • 실제 Spring MVC 패턴은 더 복잡하게 구현되어 있다.

전체적인 구조


핸들러 어댑터 : 중간에 핸들러 어댑터를 통해 다양한 종류의 컨트롤러를 호출할 수 있다.
핸들러 : 컨트롤러의 더 넓은 범위이다. 어떠한 것이든 해당 종류의 어댑터만 있으면 다 처리할 수 있다.

어댑터용 인터페이스 MyHandlerAdapter

public interface MyHandlerAdapter {
	
   boolean supports(Object handler);

	// 실제 컨트롤러를 호출하고, 그 결과로 ModelView 반환
   // 실제 컨트롤러가 ModelView를 반환하지 못하면, 어댑터가 ModelView를 직접 생성해서라도 반환
   ModelView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException, IOException;
}

boolean supports(Object handler)

  • 어댑터가 해당 컨트롤러를 처리할 수 있는지 판단하는 메서드
  • (Object hanlder) : 컨트롤러

ModelView handle(HttpServletRequest request, HttpServletResponse response, Object handler)

  • 어댑터는 실제 컨트롤러를 호출, ModelView 반환
  • 실제 컨트롤러가 ModelView를 반환하지 못할 경우, 어댑터가 ModelView를 직접 생성해서 반환
  • 어댑터를 통해서 실제 컨트롤러가 호출된다.

ControllerV3를 지원하는 어댑터

public class ControllerV3HandlerAdapter implements MyHandlerAdapter {

    @Override
    public boolean supports(Object handler) {
        return (handler instanceof ControllerV3);
    }

    @Override
    public ModelView handle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException, IOException {
    
    	// supports를 먼저 거치므로 사용 가능
        ControllerV3 controller = (ControllerV3) handler;

        Map<String, String> paramMap = createParamMap(request);
        ModelView mv = controller.process(paramMap);
        return mv;
    }

    private Map<String, String> createParamMap(HttpServletRequest request) {
        Map<String, String> paramMap = new HashMap<>();
        request.getParameterNames().asIterator()
                .forEachRemaining(paramName -> paramMap.put(paramName, request.getParameter(paramName)));
        return paramMap;
    }

}

++ 다른 컨트롤러가 들어와도, 조건에 맞게 오버라이딩해서 사용 가능하다.

FrontController

@WebServlet(name = "frontControllerServletV5", urlPatterns = "/front-controller/v5/*")
public class FrontControllerServletV5 extends HttpServlet {

    private final Map<String, Object> handlerMappingMap = new HashMap<>();
    private final List<MyHandlerAdapter> handlerAdapters = new ArrayList<>();

    public FrontControllerServletV5() {
        initHandlerMappingMap();
        initHandlerAdapters();
    }

    private void initHandlerMappingMap() {
    	// 적용할 컨트롤러들을 등록한다.
        // V3
        handlerMappingMap.put("/front-controller/v5/v3/members/new-form", new MemberFormControllerV3());
        handlerMappingMap.put("/front-controller/v5/v3/members/save", new MemberSaveControllerV3());
        handlerMappingMap.put("/front-controller/v5/v3/members", new MemberListControllerV3());

        //V4
        handlerMappingMap.put("/front-controller/v5/v4/members/new-form", new MemberFormControllerV4());
        handlerMappingMap.put("/front-controller/v5/v4/members/save", new MemberSaveControllerV4());
        handlerMappingMap.put("/front-controller/v5/v4/members", new MemberListControllerV4());
    }

    private void initHandlerAdapters() {
    	// 각 버전에 맞는 HandlerAdapter를 등록한다.
        handlerAdapters.add(new ControllerV3HandlerAdapter());
        handlerAdapters.add(new ControllerV4HandlerAdapter());
    }

    @Override
    protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		
        // 처리할 수 있는 Handler인지 검사
        Object handler = getHandler(request);
        if (handler == null) {
            response.setStatus(HttpServletResponse.SC_NOT_FOUND);
            return;
        }

        MyHandlerAdapter adapter = getHandlerAdapter(handler);

        ModelView mv = adapter.handle(request, response, handler);
        MyView view = viewResolver(mv.getViewName());
        view.render(mv.getModel(), request, response);
   
    }

    private MyHandlerAdapter getHandlerAdapter(Object handler) {
        for (MyHandlerAdapter adapter : handlerAdapters) {
            if (adapter.supports(handler)) {
                return adapter;
            }
        }
        throw new IllegalArgumentException("handler adapter is not found");
    }

    private Object getHandler(HttpServletRequest request) {
        String requestURI = request.getRequestURI();
        return handlerMappingMap.get(requestURI);
    }

    private MyView viewResolver(String viewName) {
        return new MyView("/WEB-INF/views/" + viewName + ".jsp");
    }
}

로직

  • 생성자를 통해 사용할 핸들러 매핑과 어댑터를 초기화(등록)한다.
  • getHandler(HttpServletRequest requset)를 통해 요청된 URI에 맞는 컨트롤러를 반환해준다.
  • 만약 handler == null이라면 등록되지 않은 URI이므로 404를 리턴한다.
  • 핸들러를 처리할 수 있는 어댑터 조회 : getHanlderAdapter(handler)를 통해 알맞은 핸들러 어댑터를 찾는다.
  • adapter.handle(request, response, handler);를 통해 실제 어댑터 호출
  • 그 후, render (Attribute에 저장, dispatcher.forward)

실제 스프링 MVC 패턴도 이와 유사하게 작동한다.
다만, 에노테이션기반으로 편리하게 사용할 수 있다.


'스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술 - 김영한 님' 의 강의 내용을 정리한 것입니다.

profile
백엔드 개발자

0개의 댓글