작성
·
589
·
수정됨
1
@Controller
@RequestMapping("/springmvc/v1/members/new-form")
public class SpringMemberFormControllerV1 {
public ModelAndView process() {
return new ModelAndView("new-form");
}
}
궁금해서 @RequestMapping을 클래스 레벨에 작성해보았습니다. 404 오류가 발생했는데, 추측해본 결과는 아래와 같습니다. 제가 추측한 부분이 맞는지 봐주실 수 있을까요?
/springmvc/v1/members/new-form 요청이 들어왔다.
-> 핸들러 매핑에서 SpringMemberFormControllerV1가 조회되었다.
-> SpringMemberFormControllerV1를 처리할 수 있는 핸들러 어댑터로 RequestMappingHandlerAdapter가 선택되었다.
-> DispatcherServlet이 핸들러 어댑터를 호출하였다.
-> 핸들러 어댑터가 핸들러를 호출하려고 하는데, 이때, 404 오류가 발생한다.
또는
/springmvc/v1/members/new-form 요청이 들어왔다.
-> @RequestMapping가 메서드 레벨이 아닌 클래스 레벨에 있으므로, 핸들러가 조회할 수 없다.
(그니까, RequestMappingHandlerMapping은 스프링 빈 중에서, 클래스 레벨에 @RequestMapping 또는 @Controller가 붙은 경우, 매핑 정보로 인식한다. 클래스 레벨에 이 애너테이션이 붙은 클래스는 무수히 많다. 그렇다면, 요청 URL도 참고해서 핸들러를 조회한다는 것인데, @RequestMapping이 메서드 레벨에 존재하지 않기 때문에 핸들러를 조회할 수 없다.)
답변 1
1
안녕하세요, 도토리 님. 공식 서포터즈 y2gcoder 입니다.
좀 더 확실하게 답변드리기 위해 디버깅을 해봤습니다.
(그니까, RequestMappingHandlerMapping은 스프링 빈 중에서, 클래스 레벨에 @RequestMapping 또는 @Controller가 붙은 경우, 매핑 정보로 인식한다. 클래스 레벨에 이 애너테이션이 붙은 클래스는 무수히 많다. 그렇다면, 요청 URL도 참고해서 핸들러를 조회한다는 것인데, @RequestMapping이 메서드 레벨에 존재하지 않기 때문에 핸들러를 조회할 수 없다.)
결론부터 말씀드리면 이렇게 이해하시는 게 맞습니다.
스프링 컨테이너를 시작하면서 등록된 빈들 중 @Controller나 @RequestMapping이 붙은 빈을 찾습니다.
찾은 핸들러(빈)에서 @RequestMapping이 붙은 메서드를 찾아 RequestMappingInfo 객체를 만들어줍니다. 이 RequestMappingInfo에는 @RequestMapping의 url, HTTP Method 등의 정보가 들어있습니다.
RequestMappingHandlerMapping의 상위 추상 클래스인 AbstractHandlerMethodMapping의 mappingRegistry에 찾은 RequestMappingInfo, method, handler를 등록한다.
클라이언트의 요청 때는 request 객체를 통해 path를 가져와서, mappingRegistry에서 등록한 RequestMappingInfo를 이용해 해당하는 핸들러를 찾아서 가져옵니다.
그런데 직접 디버깅해본 결과 @RequestMapping의 메서드에 주지 않았을 때는 RequestMappingInfo를 만들지 못하고, 결과적으로 해당 RequestMappingInfo가 mappingRegistry에 등록되지 않았습니다.
직접 디버깅해보실 때를 대비해 제가 디버깅 시 사용했던 break point를 알려드리겠습니다.
RequestMappingHandlerMapping.java 입니다.
AbstractHandlerMethodMapping.java 입니다.
이렇게 걸어놓고 프로젝트를 구동하시면 핸들러 매핑과정을 보실 수 있으실 거라 생각합니다.
감사합니다.