조컴퓨터

#스프링 MVC - 구조 이해 본문

공부/Spring

#스프링 MVC - 구조 이해

챠오위 2022. 1. 27. 23:15

###뇌 정리용 글###

 


두 가지 프레임워크를 비교해 본다.

 

직접 만든 프레임워크 -> 스프링 MVC 비교

- FrontController -> DispatcherServlet

- handlerMappingMap -> HandlerMapping

- MyHandlerAdapter -> HandlerAdapter

- ModelView -> ModelAndView

- viewResolver -> ViewResolver

- MyView -> View

 


DispatcherServlet 구조

org.springframework.web.servlet.DispatcherServlet

DispatcherServlet 은 스프링 MVC 의 핵심이다.

 

 

DispatcherServlet 서블릿 등록

- DispatcherServlet 도 부모 클래스에서 HttpServlet 을 상속 받아서 사용하고, 서블릿으로 동작한다.

DispatcherServlet -> FrameworkServlet -> HttpServletBean -> HttpServlet

- 스프링 부트는 DispatcherServlet 을 서블릿으로 자동 등록하면서 모든 경로(urlPattern="/") 에 대해서 매핑한다.

  - 참고) 더 자세한 경로가 우선순위가 높다. 그래서 기존에 등록한 서블릿도 함께 동작한다.

 

 


 

동작 순서

1. 핸들러 조회: 핸들러 매핑을 통해 요청 URL 에 매핑된 핸들러(컨트롤러)를 조회한다.

2. 핸들러 어댑터 조회: 핸들러를 실행할 수 있는 핸들러 어댑터를 조회한다.

3. 핸들러 어댑터 실행: 핸들러 어댑터를 실행한다.

4. 핸들러 실행: 핸들러 어댑터가 실제 핸들러를 실행한다.

5. ModelAndView 반환: 핸들러 어댑터는 핸들러가 반환하는 정보를 ModelAndView 로 변환해서 반환한다.

6. viewResolver 호출: 뷰 리졸버를 찾고 실행한다.

   - JSP 의 경우: InternalResourceViesResolver 가 자동 등록되고, 사용된다.

7. View 반환: 뷰 리졸버는 뷰의 논리 이름을 물리 이름으로 바꾸고, 렌더링 역할을 담당하는 뷰 객체를 반환한다. 

   - JSP 의 경우 InternalResourceView(JstlView) 를 반환하는데, 내부에 forward() 로직이 있다.

8. 뷰 렌더링: 뷰를 통해서 뷰를 렌더링 한다.

 

 

*주요 인터페이스 목록

- 핸들러 매핑: org.springframework.web.servlet.HandlerMapping
- 핸들러 어댑터: org.springframework.web.servlet.HandlerAdapter
- 뷰 리졸버: org.springframework.web.servlet.ViewResolver

- 뷰: org.springframework.web.servlet.View

 

 

1. 핸들러 조회

   - HandlerMapping(핸들러 매핑)

0 = RequestMappingHandlerMapping : 애노테이션 기반의 컨트롤러인 @RequestMapping에서 사용
1 = BeanNameUrlHandlerMapping : 스프링 빈의 이름으로 핸들러를 찾는다.

 

2. 핸들러 어댑터 조회

    - HandlerAdapter(핸들러 어댑터)

0 = RequestMappingHandlerAdapter : 애노테이션 기반의 컨트롤러인 @RequestMapping에서 사용
1 = HttpRequestHandlerAdapter : HttpRequestHandler 처리
2 = SimpleControllerHandlerAdapter : Controller 인터페이스(애노테이션X, 과거에 사용) 처리

*핸들러 매핑도, 핸들러 어댑터도 모두 순서대로 찾고 만약 없으면 다음 순서로 넘어간다.

 

 

@RequestMapping

가장 우선순위가 높은 핸들러 매핑과 핸들러 어댑터는

RequestMappingHandlerMapping, RequestMappingHandlerAdapter 이다.

@RequestMapping 의 앞글자를 따서 만든 이름인데, 이것이 바로 지금 스프링에서 주로 사용하는 애노테이션 기반의 컨트롤러를 지원하는 매핑과 어댑터이다. 실무에서는 99.9% 이 방식의 컨트롤러를 사용한다.

 

 


뷰 리졸버

application.properties 에 다음 코드 추가

spring.mvc.view.prefix=/WEB-INF/views/
spring.mvc.view.suffix=.jsp

뷰 리졸버 - InternalResourceViewResolver

스프링 부트는 InternalResourceViewResolver 라는 뷰 리졸버를 자동으로 등록하는데, 이때 application.properties 에 등록한 spring.mvc.view.prefix , spring.mvc.view.suffix 설정 정보를 기반으로 등록한다.

 


 

6. viewResolver 호출

획득한 논리 이름을 바탕으로 viewResolver 를 순서대로 호출한다.

 

- viewResolver 

1 = BeanNameViewResolver : 빈 이름으로 뷰를 찾아서 반환한다. (예: 엑셀 파일 생성 기능에 사용)
2 = InternalResourceViewResolver : JSP를 처리할 수 있는 뷰를 반환한다.

*뷰 리졸버 또한 모두 순서대로 찾고 만약 없으면 다음 순서로 넘어간다.

 

 

 


스프링 MVC 예제 - 실용적인 방식

실무에서는 다음과 같은 방식을 주로 사용한다.

@Controller
@RequestMapping("/springmvc/v3/members")
public class SpringMemberControllerV3 {

    private MemberRepository memberRepository = MemberRepository.getInstance();

    @GetMapping("/new-form")
    public String newForm() {
        return "new-form";
    }

    @PostMapping("/save")
    public String save(
            @RequestParam("username") String username,
            @RequestParam("age") int age,
            Model model) {

        Member member = new Member(username, age);
        memberRepository.save(member);

        model.addAttribute("member", member);
        return "save-result";
    }

    @GetMapping
    public String members(Model model) {

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

        model.addAttribute("members", members);
        return "members";
    }
}

 

Model 파라미터

save(), members() 를 보면 Model 을 파라미터로 받는 것을 확인할 수 있다. 스프링 MVC 도 이런 편의 기능을 제공한다.

 

ViewName 직접 반환

뷰의 논리 이름을 반환할 수 있다.

 

@RequestParam 사용

스프링은 HTTP 요청 파라미터를 @RequestParam 으로 받을 수 있다.

@RequestParam("username") 은 request.getParameter("username") 과 거의 같은 코드라고 생각하면 된다.

물론 GET 쿼리 파라미터, POST Form 방식을 모두 지원한다.

 

@RequestMapping -> @GetMapping, @PostMapping

@RequestMapping 은 URL 만 매칭하는 것이 아니라, HTTP Method 도 함께 구분할 수 있다.

 

 

 

참고)

1. 김영한 님의 '스프링 MVC 1편 - 백엔드 웹 개발 핵심'에서 MVC 프레임워크 만들기

 

 

 

 

'공부 > Spring' 카테고리의 다른 글

ResponseEntity  (0) 2022.01.29
프로젝트 생성과 로깅, 요청 매핑  (0) 2022.01.28
#MVC 프레임워크 만들기 - 유연한 컨트롤러 - v5  (0) 2022.01.27
#MVC 프레임워크 만들기 - v1, v2, v3, v4  (0) 2022.01.27
MVC 패턴  (0) 2022.01.26