Spring MVC 패턴을 직접 구현해보는 내용이 있어 듣고 정리해보았다.
강의에선 V1 ~ V5까지 다 구현을 하였는데, 내용이 너무 많아 V5의 내용만 포스팅할 예정이다.
V5의 핵심은 어느 종류의 컨트롤러도 유연하게 다룰 수 있다는 점이다. (끝판왕)
핸들러 어댑터 : 중간에 핸들러 어댑터를 통해 다양한 종류의 컨트롤러를 호출할 수 있다.
핸들러 : 컨트롤러의 더 넓은 범위이다. 어떠한 것이든 해당 종류의 어댑터만 있으면 다 처리할 수 있다.
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)
ModelView handle(HttpServletRequest request, HttpServletResponse response, Object handler)
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;
}
}
++ 다른 컨트롤러가 들어와도, 조건에 맞게 오버라이딩해서 사용 가능하다.
@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");
}
}
실제 스프링 MVC 패턴도 이와 유사하게 작동한다.
다만, 에노테이션기반으로 편리하게 사용할 수 있다.
'스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술 - 김영한 님' 의 강의 내용을 정리한 것입니다.