[Spring MVC] #2 프레임워크 만들기 (1) 프론트 컨트롤러 패턴

Jaeyoo (유재형)·2022년 2월 2일
1

SpringMVC

목록 보기
2/12
post-thumbnail
post-custom-banner

MVC 패턴을 이용해 직접 프레임워크 구조를 만들어 보면서 Spring MVC의 구조를 공부해보려한다.
(김영한님의 스프링 MVC 강의를 듣고 정리했다.)


프론트 컨트롤러


프론트 컨트롤러 패턴이란?

  • 각각의 컨트롤러에서 클라이언트 요청을 받는게 아니라 프론트 컨트롤러 서블릿 하나로 클라이언트 요청 받음
  • 프론트 컨트롤러가 요청에 맞는 컨트롤러를 찾아서 호출
  • 입구가 하나이므로 공통 처리 가능
  • 프론트 컨트롤러를 제외한 나머지 컨트롤러는 서블릿을 사용하지않아도 됨
    ❗️ 서블릿을 사용한 이유는 HTTP 스펙을 편하게 사용하기 위해 였다.

⚠️ Spring MVC의 DispatcherServlet이 Front Controller 패턴으로 구현되어있다.


프론트 컨트롤러 도입 버전

ControllerV1

public interface ControllerV1 {

    void process(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException;
}
  • 다형성을 활용하기위해 controller는 interface로 구현한다.
  • 서블릿과 비슷한 모양의 controller interface를 만든다.
  • 각각의 컨트롤러들은 인터페이스를 구현

❗️ 프론트 컨트롤러는 인터페이스를 호출해 구현과 관계없이 로직의 일관성을 갖는다.

회원 등록 컨트롤러

public class MemberFormController implements ControllerV1 {

    @Override
    public void process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        String viewPath = "/WEB-INF/views/new-form.jsp";
        RequestDispatcher dispatcher = request.getRequestDispatcher(viewPath);
        dispatcher.forward(request, response);
    }
}

회원 저장 컨트롤러

public class MemberSaveController implements ControllerV1 {

    MemberRepository memberRepository = MemberRepository.getInstance();

    @Override
    public void 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);
        //Model에 데이터를 보관한다.
        request.setAttribute("member", member);

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

회원 목록 컨트롤러

public class MemberListController implements ControllerV1 {

    MemberRepository memberRepository = MemberRepository.getInstance();

    @Override
    public void process(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        List<Member> members = memberRepository.findAll();

        request.setAttribute("members", members);

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

FrontControllerV1

@WebServlet(name = "frontControllerServletV1", urlPatterns = "/front-controller/v1/*")
public class FrontControllerServletV1 extends HttpServlet {

    private Map<String, ControllerV1> controllerMap = new HashMap<>();

    public FrontControllerServletV1() {
        controllerMap.put("/front-controller/v1/members/new-form", new MemberFormController());
        controllerMap.put("/front-controller/v1/members/save", new MemberSaveController());
        controllerMap.put("/front-controller/v1/members", new MemberListController());
    }

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

        String requestURI = request.getRequestURI();
        ControllerV1 controller = controllerMap.get(requestURI);
        if (controller == null){
            response.setStatus(HttpServletResponse.SC_NOT_FOUND);
            return;
        }
        controller.process(request, response);
    }
}
  • 만약 controller를 인터페이스로 구현하지않았다면 일일이 if문으로 처리해주어야함
    • 다형성을 이용해서 DIP 원칙을 지켰다.
  • @WebServlet의 urlpatterns을 보면 frontController 이기때문에 경로가 xx/v1/* 인것을 볼수있다.
    ➡️ url에 v1/ 의 아무거나 와도 frontController를 거친다는 의미
  • Map의 key와 value가 url과 Controller 이다.
    ➡️ 매핑정보를 미리 Map에 저장

더 개선해야될 부분

각 컨트롤러에서 뷰로 이동하는 부분에 코드 중복이 있다.

String viewPath = "/WEB-INF/views/xxx.jsp";
RequestDispatcher dispatcher = request.getRequestDispatcher(viewPath);
dispatcher.forward(request, response);
profile
기록과 반복
post-custom-banner

0개의 댓글