Spring 49

[Spring MVC] 프론트 컨트롤러 - 어댑터 추가 (1)

유연한 컨트롤러 - V5 만약 어떤 개발자는 ControllerV3 방식으로 개발하고 싶고, 어떤 개발자는 ControllerV4 방식으로 개발하고 싶다면 어떻게 해야 할까? public interface ControllerV3 { ModelView process(Map paramMap); } public interface ControllerV4 { String process(Map paramMap, Map model); } 어댑터 패턴 지금까지 우리가 개발한 프론트 컨트롤러는 한 가지 방식의 컨트롤러 인터페이스만 사용할 수 있다. ControllerV3 , ControllerV4는 완전히 다른 인터페이스이다. 따라서 호환이 불가능하다. 마치 v3는 110v이고, v4는 220v 전기 콘센트 같은 것이다..

[Spring MVC] 프론트 컨트롤러 - 단순하고 실용적인 컨트롤러

단순하고 실용적인 컨트롤러 - V4 앞서 만든 v3 컨트롤러는 서블릿 종속성을 제거하고 뷰 경로의 중복을 제거하는 등, 잘 설계된 컨트롤러이다. 그런데 실제 컨트톨러 인터페이스를 구현하는 개발자 입장에서 보면, 항상 ModelView 객체를 생성하고 반환해야 하는 부분이 조금은 번거롭다. 좋은 프레임워크는 아키텍처도 중요하지만, 그와 더불어 실제 개발하는 개발자가 단순하고 편리하게 사용할 수 있어야 한다. 소위 실용성이 있어야 한다. 이번에는 v3를 조금 변경해서 실제 구현하는 개발자들이 매우 편리하게 개발할 수 있는 v4 버전을 개발해 보자. V4 구조 기본적인 구조는 V3와 같다. 대신에 컨트롤러가 ModelView를 반환하지 않고, ViewName 만 반환한다. ControllerV4 public ..

[Spring MVC] 프론트 컨트롤러 - Model 추가

Model 추가 - V3 서블릿 종속성 제거 컨트롤러 입장에서 HttpServletRequest, HttpServletResponse이 꼭 필요할까? @Override public MyView process(HttpServletRequest request, HttpServletResponse response) 요청 파라미터 정보는 자바의 Map으로 대신 넘기도록 하면 지금 구조에서는 컨트롤러가 서블릿 기술을 몰라도 동작할 수 있다. 그리고 request 객체를 Model로 사용하는 대신에 별도의 Model 객체를 만들어서 반환하면 된다. 우리가 구현하는 컨트롤러가 서블릿 기술을 전혀 사용하지 않도록 변경해 보자. 이렇게 하면 구현 코드도 매우 단순해지고, 테스트 코드 작성이 쉽다. 뷰 이름 중복 제거 컨..

[Spring MVC] 프론트 컨트롤러 - View 분리

View 분리 - V2 String viewPath = "/WEB-INF/views/new-form.jsp"; RequestDispatcher dispatcher = request.getRequestDispatcher(viewPath); dispatcher.forward(request, response); 앞서 프론트 컨트롤러 도입을 통해, 모든 컨트롤러의 입구를 하나로 만들어보았다. 하지만 위의 코드처럼 모든 컨트롤러에서 뷰로 이동하는 부분에 중복이 있고, 깔끔하지 않다. 이 부분도 프론트 컨트롤러를 통해 분리하여 공통 처리해 보도록 하자. V2 구조 앞의 V1 구조와 달라진 점은 MyView의 존재 여부이다. 프론트 컨트롤러를 통해 각 컨트롤러를 호출하게 되고 해당 컨트롤러는 자신의 로직을 수행한 뒤..

[Spring MVC] 프론트 컨트롤러 패턴

프론트 컨트롤러 패턴 소개 프론트 컨트롤러 도입 전 요청이 들어오는 입구가 여러 곳 프론트 컨트롤러 도입 후 FrontController 패턴 특징 프론트 컨트롤러 서블릿 하나로 클라이언트의 요청을 받음 프론트 컨트롤러가 요청에 맞는 컨트롤러를 찾아서 호출 요청이 들어오는 입구가 하나 공통 처리 가능 프론트 컨트롤러를 제외한 나머지 컨트롤러는 서블릿을 사용하지 않아도 됨 스프링 웹 MVC와 프론트 컨트롤러 스프링 웹 MVC의 핵심은 FrontController이다. 스프링 웹 MVC의 DispatcherServlet이 FrontController 패턴으로 구현되어 있다. 프론트 컨트롤러 도입 - V1 프론트 컨트롤러를 단계적으로 도입해 보자. 이번 목표는 기존 코드를 최대한 유지하면서, 프론트 컨트롤러를..

[Spring MVC] MVC 패턴 - 한계

MVC 패턴 - 한계 MVC 패턴을 적용한 덕분에 컨트롤러의 역할과 뷰를 렌더링 하는 역할을 명확하게 구분할 수 있다. 특히 뷰는 화면을 그리는 역할에 충실한 덕분에 코드가 깔끔하고 직관적이다. 단순하게 모델에서 필요한 데이터를 꺼내고 화면을 만들면 된다. 그런데 컨트롤러는 딱 봐도 중복이 많고 필요하지 않은 코드들도 많이 보인다. MVC 컨트롤러의 단점 포워드 중복 View로 이동하는 코드가 항상 중복 호출되어야 한다. 물론 이 부분을 메서드로 공통화해도 되지만, 해당 메서드도 항상 직접 호출해야 한다. RequestDispatcher dispatcher = request.getRequestDispatcher(viewPath); dispatcher.forward(request, response); Vi..

[Spring MVC] MVC 패턴 - 적용

MVC 패턴 - 적용 서블릿을 컨트롤러로 사용하고, JSP를 뷰로 사용해서 MVC 패턴을 적용해 보자. Model은 HttpServletRequest 객체를 사용한다. request는 내부에 데이터 저장소를 가지고 있는데, request.setAttribute() , request.getAttribute()를 사용하면 데이터를 보관하고 조회할 수 있다. 회원 등록 회원 등록 폼 - 컨트롤러 hello.servlet.web.servletmvc.MvcMemberFormServlet package hello.servlet.web.servletmvc; import javax.servlet.RequestDispatcher; import javax.servlet.ServletException; import java..

[Spring MVC] MVC 패턴 - 개요

MVC 패턴 - 개요 너무 많은 역할 하나의 서블릿이나 JSP만으로 비즈니스 로직과 뷰 렌더링까지 모두 처리하게 되면, 너무 많은 역할을 하게 되고 결과적으로 유지보수가 어려워진다. 비즈니스 로직을 호출하는 부분에 변경이 발생해도 해당 코드를 손대야 하고, UI를 변경할 일이 있어도 비즈니스 로직이 함께 있는 해당 파일을 수정해야 한다. 변경의 라이프 사이클 진짜 문제는 둘 사이에 변경의 라이프 사이클이 다르다는 점이다. 예를 들어서 UI를 일부 수정하는 일과 비즈니스 로직을 수정하는 일은 각각 다르게 발생할 가능성이 매우 높고 대부분 서로에게 영향을 주지 않는다. 이렇게 변경의 라이프 사이클이 다른 부분을 하나의 코드로 관리하는 것은 유지보수하기 좋지 않다. (물론 UI가 많이 변하면 함께 변경될 가능..

[Spring MVC] JSP로 회원 관리 웹 애플리케이션 만들기

JSP로 회원 관리 웹 애플리케이션 만들기 JSP 라이브러리 추가 JSP를 사용하려면 먼저 다음 라이브러리를 추가해야 한다. 스프링 부트 3.0 미만 build.gradle 에 추가 //JSP 추가 시작 implementation 'org.apache.tomcat.embed:tomcat-embed-jasper' implementation 'javax.servlet:jstl' //JSP 추가 끝 스프링 부트 3.0 이상 build.gradle 에 추가 //JSP 추가 시작 implementation 'org.apache.tomcat.embed:tomcat-embed-jasper' implementation 'jakarta.servlet:jakarta.servlet-api' //스프링부트 3.0 이상 imp..

[Spring MVC] 서블릿으로 회원 관리 웹 애플리케이션 만들기

서블릿으로 회원 관리 웹 애플리케이션 만들기 이제 본격적으로 서블릿으로 회원 관리 웹 애플리케이션을 만들어보자. 가장 먼저 서블릿으로 회원 등록 HTML 폼을 제공해 보자. MemberFormServlet - 회원 등록 폼 package hello.servlet.web.servlet; import hello.servlet.domain.member.MemberRepository; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.serv..