minOS

MVC 프레임워크 만들기 - 프론트 컨트롤러 Ver 2 본문

TIL/김영한의 스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술

MVC 프레임워크 만들기 - 프론트 컨트롤러 Ver 2

minOE 2024. 6. 29. 15:51
728x90

View 분리 - Ver 2

String viewPath = "/WEB-INF/views/new-form.jsp";
 RequestDispatcher dispatcher = request.getRequestDispatcher(viewPath);
dispatcher.forward(request, response);


모든 컨트롤러에 view로 이동하는 부분에 중복이 있다.
이 부분을 분리하기 위해 별도의 view를 처리하는 객체를 만들어야 한다.

VER 2 구조


MyView
이 코드를 보고 어떻게 사용할지 감이 안온다  ->  해당 코드가  없다고 생각하고 왜 이렇게 만들었을지 생각해보자 !!

public class MyView {
    String viewPath;

    public MyView(String urlPath) {
        this.viewPath = urlPath;
    }

    public void render(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        RequestDispatcher dispatcher = request.getRequestDispatcher(viewPath);
        dispatcher.forward(request, response);

    }
}


ver1에서 ControllerV1은 리턴 타입이 void인 메서드를 선언한 인터페이스이지만 보다시피 ver2은 ControllerV2 인터페이스의 메서드는 리턴 타입이 MyView인 인터페이스다.

public interface ControllerV2  {
    MyView process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException;
}

 

MemberFormControllerV2 - 회원 등록 폼

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");
    }
}


이렇게 컨트롤러가 MyView 객체를 생성하고 거기에 url경로를 넣으면 MyView 가 반환된다.

@WebServlet(name="frontControllerServletV2",urlPatterns = "/front-controller/v2/*")
public class FrontControllerServletV2 extends HttpServlet {
//key  == url , value = controller
    private Map<String, ControllerV2 > controllerMap = new HashMap<>();

    public FrontControllerServletV2() {
        controllerMap.put("/front-controller/v2/members/new-form", new MemberFormControllerV2());
        controllerMap.put("/front-controller/v2/members/save", new MemberSaveControllerV2());
        controllerMap.put("/front-controller/v2/members", new MemberListControllerV2());
    }

    @Override
    protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        System.out.println("FrontControllerServletV1.service");

        String requestURI = request.getRequestURI();
        ControllerV2 controller = controllerMap.get(requestURI);
        if (controller == null){
            response.setStatus(HttpServletResponse.SC_NOT_FOUND);
            return;
        }
        MyView myView = controller.process(request, response);
        myView.render(request,response);


    }
}

 

controller.process(request, response);

controller.process(request, response);는 MyView 객체를 반환하고, 이후 myView.render(request, response);에서 해당 MyView 객체의 render 메서드를 호출하여 뷰를 렌더링합니다.

프론트 컨트롤러의 도입으로 `MyView` 객체의 `render()` 를 호출하는 부분을 모두 일관되게 처리할 수 있다. 각각의 컨트롤러는 MyView 객체를 생성만 해서 반환하면 된다.




김영한 강사님이 알려주지 않았으면 어떻게 중복을 제거할지 .. 김영한 강사님이 이렇게 중복을 제거한 이유 생각하자  

728x90