조컴퓨터

MVC 패턴 본문

공부/Spring

MVC 패턴

챠오위 2022. 1. 26. 12:20

디자인패턴은 코딩을 하는 사람들이 효율적으로 만들어지는 일련의 패턴들을 모아서 정형화 시켜놓은 것을 말한다.

 

MVC 패턴이 탄생하게 된 계기 또한 하나의 서블릿, 순수 JSP 만으로 비즈니스 로직과 뷰 렌더링까지 모두 처리하다 보니 각각의 페이지들이 너무나 많은 역할을 하게 되고, 결과적으로 유지보수가 어려운 데에 있어 생겨났다. 

 

그리고 MVC 패턴은 라이프 사이클 관리를 위해 탄생했다. 예를 들어, UI 를 일부 수정하는 일과 비즈니스 로직을 수정하는 일은 각각 다르게 발생할 가능성이 매우 높은데에 비해 대부분의 케이스에 있어서 서로에게 영향을 주지 않는다. 이렇게 변경에 대한 라이프 사이클이 다른 부분을 하나의 코드로 관리하는 것은 유지보수에 있어서 좋지 않다.

 

MVC 패턴은 JSP 기능을 살려주는 역할 또한 한다. JSP 와 같은 뷰 템플릿은 화면을 렌더링하는 데 있어서 최적화 되어 있기 때문에 이 부분의 업무만 담당하게 하는 것이 가장 효과적이다. 

 

 

MVC (Model View Controller)

MVC 패턴은 하나의 서블릿이나, JSP 로 처리하던 코드를 컨트롤러(Controller) 와 뷰(View) 영역으로 서로 역할을 나눈 것을 말한다. 웹 애플리케이션은 보통 이 MVC 패턴을 사용한다.

 

컨트롤러: HTTP 요청을 받아서 파라미터를 검증하고, 비즈니스 로직을 실행한다. 그리고 뷰에 전달할 결과 데이터를 조회해서 모델에 담는다.

모델: 뷰에 출력할 데이터를 담아둔다. 뷰가 필요한 데이터를 모두 모델에 담아서 전달해주는 덕분에 뷰는 비즈니스 로직이나 데이터 접근을 몰라도 되고, 화면을 렌더링하는 일에 집중할 수 있다.

: 모델에 담겨 있는 데이터를 사용해서 화면을 그리는 일에 집중한다. 여기서는 HTML 을 생성하는 부분을 말한다.

 

 

참고) 

컨트롤러에 비즈니스 로직을 둘 수 있지만, 이렇게 되면 컨트롤러에 너무 많은 역할이 주어진다. 그래서 일반적으로 비즈니스 로직은 서비스(Service) 계층을 별도로 만들어 처리한다. 그리고 컨트롤러는 비즈니스 로직이 있는 서비스를 호출하는 역할을 맡는다. 참고로 비즈니스 로직을 변경하면 비즈니스 로직을 호출하는 컨트롤러의 코드도 변경될 수 있다.

 

 

 

MVC 패턴은 크게 MVC 1 패턴과, 스프링이 채택한 MVC 2 패턴으로 나눌 수 있다. 

 

MVC1

 

MVC2 - 서비스, 리포지토리가 생략되어 있는 그림
출처) 김영한 님의 '스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술'

 

 

MVC 패턴 - 적용

서블릿을 컨트롤러로 사용하고, JSP 를 뷰로 사용해서 MVC 패턴을 적용해보자.

Model 은 HttpServletRequest 객체를 사용한다. request 는 내부에 데이터 저장소를 가지고 있는데, request.setAttribute(), request.getAttribute() 를 사용하면 데이터를 보관하고, 조회할 수 있다.

 

 

회원등록 폼 - 컨트롤러

@WebServlet(name = "mvcMemberFormServlet", urlPatterns = "/servlet-mvc/members/new-form")
public class MvcMemberFormServlet extends HttpServlet {

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

- dispatcher.forward(): 다른 서블릿이나 JSP 로 이동할 수 있는 기능이다. 서버 내부에서 다시 호출이 발생한다.

- /WEB-INF: 이 경로 안에 JSP 파일이 있으면 외부에서 직접 JSP 를 호출할 수 없다. 우리가 기대하는 것은 항상 컨트롤러를 통해서 JSP 를 호출하는 것이다.

- redirect vs forward

리다이렉트는 실제 클라이언트(웹 브라우저)에 응답이 나갔다가, 클라이언트가 redirect 경로로 다시 요청한다. 따라서 클라이언트가 인지할 수 있고, URL 경로도 실제로 변경된다. 반면에 포워드는 서버 내부에서 일어나는 호출이기 때문에 클라이언트가 전혀 인지하지 못한다.

 

 

*상대경로

<!-- 상대경로 사용, [현재 URL 이 속한 계층 경로 + /save] -->
<form action="save" method="post">

form 의 action 을 보면 절대경로가 아니라 상대경로인 것을 확인할 수 있다. 이렇게 상대경로를 사용하면 폼 전송시 현재 URL 이 속한 경로 + save 가 호출된다.

현재 계층 경로: /servlet-mvc/members

결과: /servlet-mvc/members/save

 

 

회원 저장 - 컨트롤러

@WebServlet(name = "mvcMemberSaveServlet", urlPatterns = "/servlet-mvc/members/save")
public class MvcMemberSaveServlet extends HttpServlet {

    private MemberRepository memberRepository = MemberRepository.getInstance();

    @Override
    protected void service(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);
    }
}

- HttpServletRequest 를 Model 로 사용한다.

- request 가 제공하는 setAttribute() 를 사용하면 request 객체에 데이터를 보관해서 뷰에 전달할 수 있다.

- 뷰는 request.getAttribute() 를 사용해서 데이터를 꺼내면 된다.

 

 

회원 저장 - 뷰

<%--
    <li>id=<%=((Member)request.getAttribute("member")).getId()%></li>
    <li>username=<%=((Member)request.getAttribute("member")).getUsername()%></li>
    <li>age=<%=((Member)request.getAttribute("member")).getAge()%></li>
--%>

<li>id=${member.id}</li>
<li>username=${member.username}</li>
<li>age=${member.age}</li>

- JSP 는 ${} 문법을 제공하는데, 이 문법을 사용하면 request 의 attribute 에 담긴 데이터를 편리하게 조회할 수 있다.

 


MVC 덕분에 컨트롤러 로직과 뷰 로직이 확실하게 분리된 것을 확인할 수 있다. 이후에 화면 수정이 필요하면 뷰 로직을 수정하면 된다.

 

 

회원 목록 조회 - 컨트롤러

List<Member> members = memberRepository.findAll();

request 객체를 사용해서 List<Member> members 를 모델에 보관했다.

 

 

회원 목록 조회 - 뷰

// <c: forEach> 기능을 사용하기 위해 다음을 선언해야 한다.
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
...

<c:forEach var="item" items="${members}">
    <tr>
        <td>${item.id}</td>
        <td>${item.username}</td>
        <td>${item.age}</td>
    </tr>
</c:forEach>

모델에 담아둔 members 를 JSP 가 제공하는 taglib 기능을 사용해서 반복하면서 출력했다. members 리스트에서 member 를 순서대로 꺼내서 item 변수에 담고, 출력하는 과정을 반복한다.

 

 

MVC 패턴 - 한계

MVC 패턴을 적용한 덕분에 컨트롤러의 역할과 뷰를 렌더링하는 역할을 명확하게 구분할 수 있다. 특히 뷰는 화면을 그리는 역할에 충실한 덕분에, 코드가 깔끔하고 직관적이다. 단순하게 모델에서 필요한 데이터를 꺼내고, 화면을 만들면 된다. 그런데 컨트롤러는 중복이 많고, 필요하지 않은 코드들도 많아 보인다.

 

 

*포워드 중복

RequestDispatcher dispatcher = request.getRequestDispatcher(viewPath);
dispatcher.forward(request, response);

 

*viewPath 중복

String viewPath = "/WEB-INF/views/new-form.jsp";

- prefix: /WEB-INF/views/

- suffix: .jsp

그리고 만약 jsp 가 아닌 thymeleaf 같은 다른 뷰로 변경한다면 전체 코드를 다 변경해야 한다.

 

 

*사용하지 않는 코드

HttpServletRequest request, HttpServletResponse response

HttpServletResponse 의 경우 현재 코드에서 잘 사용되지 않는다.

그리고 이런 HttpServletRequest, HttpServletResponse 를 사용하는 코드는 테스트 케이스를 작성하기도 어렵다.

 

 

*공통 처리가 어렵다.

기능이 복잡해질수록 컨트롤러에서 공통으로 처리해야 하는 부분이 점점 더 많아질 것이다. 단순히 공통 기능을 메서드로 정리하면 될 것 같지만, 해당 메서드를 항상 호출해야 하고, 실수로 호출하지 않는 경우에는 문제가 될 것이다. 그리고 호출하는 것 자체도 중복이다.

 

 

*정리하면 공통 처리가 어렵다는 문제가 있다.

이 문제를 해결하려면 컨트롤러 호출 전에 먼저 공통 기능을 처리해야 한다. 프론트 컨트롤러(Front Controller) 패턴을 도입하면 이런 문제를 깔끔하게 해결할 수 있다. 

스프링 MVC 의 핵심도 바로 이 프론트 컨트롤러에 있다.

 

 

 

 

참고)

1. 김영한 님의 '스프링 MVC 1편 - 백엔드 웹 개발 핵심'에서 MVC 패턴 - 개요

2. [Spring] Spring의 MVC 패턴과 MVC1과 MVC2 비교 | ChanBLOG (chanhuiseok.github.io)

 

[Spring] Spring의 MVC 패턴과 MVC1과 MVC2 비교

컴퓨터/IT/알고리즘 정리 블로그

chanhuiseok.github.io

3. 코끼리를 냉장고에 넣는 방법 :: [서블릿/JSP] 표현언어(EL)에서 ${}과 #{} 표기법의 차이 (tistory.com)

 

[서블릿/JSP] 표현언어(EL)에서 ${}과 #{} 표기법의 차이

이전글 [서블릿/JSP] 표현 언어(EL) 기본 사용법 및 자료형 설명 표현언어(EL)에서 ${}과 #{} 표기법의 차이 표현언어(Expression Language)에는 ${} 와 #{} 두가지 표기법이 있습니다. #{}표기법의 경우 JSP 2.1.

dololak.tistory.com