[스프링MVC 22] MVC 프레임워크 만들기: 프론트 컨트롤러 (FrontController) (tistory.com)
모든 컨트롤러에서 뷰 로직(뷰로 이동하는 부분)에 중복이 있다.
String viewPath = "/WEB-INF/views/new-form.jsp";
RequestDispatcher dispatcher = req.getRequestDispatcher(viewPath);
dispatcher.forward(req, resp);
이 부분을 깔끔하게 분리하기 위해 별도로 뷰를 처리하는 객체(MyView)를 만들자
- MyView
/**뷰를 처리하는 객체*/
//뷰로 이동하는 부분(중복코드) 분리
public class MyView {
private String viewPath;
public MyView(String viewPath){
this.viewPath =viewPath;
}
public void render(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
RequestDispatcher dispatcher = req.getRequestDispatcher(viewPath);
dispatcher.forward(req, resp);
}
}
- 컨트롤러 인터페이스
- 컨트롤러가 뷰를 반환하는 특징이 있다. (MyView로 반환)
public interface ControllerV2 {
MyView process(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException;
}
- 회원 등록 폼
/**회원 등록 폼*/
public class MemberFormControllerV2 implements ControllerV2 {
@Override
public MyView process(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
return new MyView("/WEB-INF/views/new-form.jsp");
}
}
- 이제 각 컨트롤러는 복잡한 dispatcher.forward() 를 직접 생성해서 호출하지 않아도 된다. 단순히 MyView 객체를 생성하고 거기에 뷰 이름만 넣고 반환하면 된다.
- ControllerV1을 구현한 클래스와 ControllerV2를 구현한 클래스를 비교해보면, 이 부분의 중복이 확실하게 제거된 것을 확인할 수 있다.
- 회원 저장
/**회원 저장*/
public class MemberSaveControllerV2 implements ControllerV2 {
private MemberRepository memberRepository = MemberRepository.getInstance();
@Override
public MyView process(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String username = req.getParameter("username");
int age = Integer.parseInt(req.getParameter("age"));
Member member = new Member(username, age);
memberRepository.save(member);
req.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 req, HttpServletResponse resp) throws ServletException, IOException {
List<Member> members = memberRepository.findAll();
req.setAttribute("members", members);
return new MyView("/WEB-INF/views/members.jsp");
}
}
- 프론트 컨트롤러
@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 MemberFormControllerV2());
controllerMap.put("/front-controller/v2/members/save", new MemberSaveControllerV2());
controllerMap.put("/front-controller/v2/members", new MemberListControllerV2());
}
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String requestURI = req.getRequestURI();
ControllerV2 controller = controllerMap.get(requestURI);
if (controller == null) {
resp.setStatus(HttpServletResponse.SC_NOT_FOUND);
return;
}
MyView view = controller.process(req, resp);
view.render(req, resp);
}
}
- ControllerV2의 반환 타입이 MyView이므로 프론트 컨트롤러는 컨트롤러의 호출 결과로 MyView를 반환 받는다. 그리고 view.render() 를 호출하면 forward 로직을 수행해서 JSP가 실행된다.
- 프론트 컨트롤러의 도입으로 MyView 객체의 render()를 호출하는 부분을 모두 일관되게 처리할 수 있다. 각각의 컨트롤러는 MyView 객체를 생성만 해서 반환하면 된다.
728x90
반응형
'Spring Tutorial' 카테고리의 다른 글
[스프링MVC 25] MVC 프레임워크 만들기: 단순하고 실용적인 컨트롤러 (0) | 2023.07.31 |
---|---|
[스프링MVC 24] MVC 프레임워크 만들기: Model 추가 (*중요) (0) | 2023.07.30 |
[스프링MVC 22] MVC 프레임워크 만들기: 프론트 컨트롤러 (FrontController) (0) | 2023.07.28 |
[스프링MVC 21] MVC 패턴의 한계 (0) | 2023.07.27 |
[스프링MVC 20] MVC 패턴으로 회원 관리 웹 애플리케이션 (0) | 2023.07.27 |