View 도입
앞서 언급한 중복 문제를 극복하기 위해 View를 도입한다. 이전 구조와는 다르게 컨트롤러가 바로 JSP로 포워딩하지 않고 View 객체를 반환한다.

View 클래스
public class MyView {
private String viewPath;
public MyView(String viewPath) {
this.viewPath = viewPath;
}
public void render(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
RequestDispatcher dispatcher = request.getRequestDispatcher(viewPath);
dispatcher.forward(request, response);
}
}
JSP 경로를 생성자 매개변수로 전달받아 필드에 저장하고, render() 메서드가 호출되면 해당 경로의 JSP로 포워딩한다.
ControllerV2 - 인터페이스
public interface ControllerV2 {
MyView process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException;
}
ControllerV2 - 구현체
public class MemberFormControllerV2 implements ControllerV2 {
@Override
public MyView process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
return new MyView("/WEB-INF/views/new-form.jsp");
}
}
public class MemberSaveControllerV2 implements ControllerV2 {
private MemberRepository memberRepository = MemberRepository.getInstance();
@Override
public MyView process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String username = request.getParameter("username");
int age = Integer.parseInt(request.getParameter("age"));
Member member = new Member(username, age);
memberRepository.save(member);
request.setAttribute("member", member);
return new MyView("/WEB-INF/views/save-result.jsp");
}
}
public class MemberListControllerV2 implements ControllerV2 {
private MemberRepository memberRepository = MemberRepository.getInstance();
@Override
public MyView process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
List<Member> members = memberRepository.findAll();
request.setAttribute("members", members);
return new MyView("/WEB-INF/views/members.jsp");
}
}
이제 각 컨트롤러는 dispatcher.forward()를 반복해서 호출하지 않아도 된다. 단순히 JSP의 경로를 전달하여 MyView객체를 생성하고, 반환하면 JSP로 포워딩하는 역할은 MyView 객체가 책임지기 때문이다.
FrontControllerServletV2 - 프론트 컨트롤러
@WebServlet(name = "frontControllerServletV2", urlPatterns = "/front-controller/v2/*")
public class FrontControllerServletV2 extends HttpServlet {
private Map<String, ControllerV2> controllerMap = new HashMap<>();
public FrontControllerServletV2() {
controllerMap.put("/front-controller/v2/members/new-form", new MemberSaveControllerV2());
controllerMap.put("/front-controller/v2/members/save", new MemberFormControllerV2());
controllerMap.put("/front-controller/v2/members", new MemberListControllerV2());
}
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String requestURI = request.getRequestURI();
ControllerV2 controllerV2 = controllerMap.get(requestURI);
if (controllerV2 == null) {
response.setStatus(HttpServletResponse.SC_NOT_FOUND);
return;
}
MyView myView = controllerV2.process(request, response);
myView.render(request, response);
}
}
컨트롤러의 process() 메서드의 반환타입이 MyView이므로 프론트 컨트롤러는 컨트롤러의 호출결과로 MyView 객체를 반환받는다. 그리고 MyView의 render() 메서드를 호출하면 forward 로직을 수행해서 JSP가 실행된다.
'Spring' 카테고리의 다른 글
| [Spring] Spring MVC : MVC 프레임워크 구현 - ModelView 개선 (0) | 2024.01.17 |
|---|---|
| [Spring] Spring MVC : MVC 프레임워크 구현 - Model 도입 (0) | 2024.01.15 |
| [Spring] Spring MVC : MVC 프레임워크 구현 - Front Controller 도입 (0) | 2023.12.06 |
| [Spring] Spring MVC : Servlet 기반의 애플리케이션 - MVC 패턴을 적용한 Servlet (1) | 2023.11.25 |
| [Spring] Spring MVC : Servlet 기반의 애플리케이션 - JSP를 적용한 요청 처리 서블릿 구현 (0) | 2023.10.03 |