스프링 MVC - 1편 - MVC 프레임워크 만들기 - 단순하고 실용적인 컨트롤러 (4)
인프런 스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술편을 학습하고 정리한 내용 입니다.
앞서 만든 v3 컨트롤러는 서블릿 종속성을 제거하고 뷰 경로의 중복을 제거하는 등, 잘 설계된 컨트롤러 이다.
그런데 실제 컨트롤러 인터페이스를 구현하는 개발자 입장에서 보면, 항상 ModelView 객체를 생성하고 반환해야 하는 부분이 조금은 번거롭다.
좋은 프레임워크는 아키텍처도 중요하지만, 그와 더불어 실제 개발하는 개발자가 단순하고 편리하게 사용할 수 있어야 한다.
소위 실용성이 있어야 한다.
이번에는 v3를 조금 변경해서 실제 구현하는 개발자들이 매우 편리하게 개발할 수 있는 v4 버전을 개발해 보자.

- 기본적인 구조는 V3와 같다. 대신에 컨트롤러가
ModelView를 반환하지 않고,ViewName만 반환 한다.
먼저 컨트롤러 인터페이스를 만들자
hello.servlet.web.frontcontroller.v4.ControllerV4
1
2
3
public interface ControllerV4 {
String process(Map<String, String> paramMap, Map<String, Object> model);
}
이젠 ModelView가 리턴이 아니라 String이 리턴이다.
컨트롤러 3개 만들자.
hello.servlet.web.frontcontroller.v4.controller.MemberFormControllerV4 - 멤버 폼1 2 3 4 5 6
public class MemberFormControllerV4 implements ControllerV4 { @Override public String process(Map<String, String> paramMap, Map<String, Object> model) { return "new-form"; } }
어마어마하게 간단해 졌다. 폼의 이름만 전달하면 되니깐.
1
2
3
4
@Override
public ModelView process(Map<String, String> paramMap) {
return new ModelView("new-form");
}
이게 v3였고
1
2
3
4
@Override
public MyView process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
return new MyView("/WEB-INF/views/new-form.jsp");
}
이게 v2 였다.
점점 짧아지는걸 볼 수 있다.
hello.servlet.web.frontcontroller.v4.controller.MemberSaveControllerV4 - 멤버 저장1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
public class MemberSaveControllerV4 implements ControllerV4 { private MemberRepository memberRepository = MemberRepository.getInstance(); @Override public String process(Map<String, String> paramMap, Map<String, Object> model) { String username = paramMap.get("username"); int age = Integer.parseInt(paramMap.get("age")); Member member = new Member(username, age); memberRepository.save(member); model.put("member", member); return "save-result"; } }
여기도 크게 달라진 건 없는데, 모델에 멤버를 담고 바로 뷰페이지 이름을 넘겨버리고 끝난다.
hello.servlet.web.frontcontroller.v4.controller.MemberListControllerV4 - 멤버 목록1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
public class MemberSaveControllerV4 implements ControllerV4 { private MemberRepository memberRepository = MemberRepository.getInstance(); @Override public String process(Map<String, String> paramMap, Map<String, Object> model) { String username = paramMap.get("username"); int age = Integer.parseInt(paramMap.get("age")); Member member = new Member(username, age); memberRepository.save(member); model.put("member", member); return "save-result"; } }
이렇게 3가지 컨트롤러를 만들었다.
이제 마지막으로 프론트 컨트롤러를 만들자.
hello.servlet.web.frontcontroller.v4.FrontControllerServletV4 - 프론트 컨트롤러
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
@WebServlet(name = "frontControllerServletV4", urlPatterns = "/front-controller/v4/*")
public class FrontControllerServletV4 extends HttpServlet {
private Map<String, ControllerV4> controllerMap = new HashMap<>();
public FrontControllerServletV4() {
controllerMap.put("/front-controller/v4/members/new-form", new MemberFormControllerV4());
controllerMap.put("/front-controller/v4/members/save", new MemberSaveControllerV4());
controllerMap.put("/front-controller/v4/members", new MemberListControllerV4());
}
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String requestURI = request.getRequestURI();
ControllerV4 controller = controllerMap.get(requestURI);
if (controller == null) {
response.setStatus(HttpServletResponse.SC_NOT_FOUND);
return;
}
// paramMap
Map<String, String> paramMap = createParamMap(request);
Map<String, Object> model = new HashMap<>(); // 추가
String viewName = controller.process(paramMap, model);
MyView view = viewResolver(viewName);
view.render(model, request, response);
}
private MyView viewResolver(String viewName) {
return new MyView("/WEB-INF/views/" + viewName + ".jsp");
}
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;
}
}
v3 프론트 컨트롤러 에서 크게 바뀐 건 없고, 이제
1
2
3
4
5
6
Map<String, Object> model = new HashMap<>(); // 추가
String viewName = controller.process(paramMap, model);
MyView view = viewResolver(viewName);
view.render(model, request, response);
모델 객체를 만들어서 뷰이름을 리턴 받고 viewResolver로 완전한 경로를 만든 후에
렌더 해주면 끝난다.



잘 작동한다.
정리
이번 버전의 컨트롤러는 매우 단순하고 실용적이다. 기존 구조에서 모델을 파라미터로 넘기고, 뷰의 논리 이름을 반환한다는 작은 아이디어를 적용했을 뿐인데, 컨트롤러를 구현하는 개발자 입장에서 보면 이제 군더더기 없는 코드를 작성할 수 있다.
또한 중요한 사실은 여기까지 한번에 온 것이 아니라는 것이다. 프레임워크가 점진적으로 발전하는 과정 속에서 이런 방법도 찾을 수 있었다.
프레임워크나 공통 기능이 수고로워야 사용하는 개발자가 편리해진다.
댓글남기기