View 도입

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

출처 : 김영한. 스프링 MVC 1편 강의자료. 인프런(https://www.inflearn.com/course/스프링-mvc-1)

 

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가 실행된다.

+ Recent posts