-
[김영한 스프링] 19. 스프링 MVC 구조 이해 - 시작하기 & 컨트롤러 통합 & 실용적인 방식Spring/스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술 2023. 7. 6. 04:13
스프링 MVC - 시작하기
스프링이 제공하는 컨트롤러는 애노테이션 기반으로 동작해서, 매우 유연하고 실용적이다. 과거에는 자바 언어에 애노테이션이 없기도 했고, 스프링도 처음부터 이런 유연한 컨트롤러를 제공한 것은 아니다.
@RequestMapping
스프링은 애노테이션을 활용한 매우 유연하고, 실용적인 컨트롤러를 만들었는데 이것이 바로 @RequestMapping 애노테이션을 사용하는 컨트롤러이다.
과거에는 스프링 프레임워크가 MVC 부분이 약해서 스프링을 사용하더라도 MVC 웹 기술은 스트럿츠 같은 다른 프레임워크를 사용했었다. 그런데 @RequestMapping 기반의 애노테이션 컨트롤러가 등장하면서, MVC 부분도 스프링의 완승으로 끝이 났다.
- RequestMappingHandlerMapping
- RequestMappingHandlerAdapter
가장 우선순위가 높은 핸들러 매핑과 핸들러 어댑터는 RequestMappingHandlerMapping, RequestMappingHandlerAdapter이다.
@RequestMapping의 앞글자를 따서 만든 이름인데, 이것이 바로 지금 스프링에서 주로 사용하는 애노테이션 기반의 컨트롤러를 지원하는 핸들러 매핑과 어댑터이다. 실무에서는 99.9% 이 방식의 컨트롤러를 사용한다.
SpringMemberFormControllerV1 - 회원 등록 폼
package hello.servlet.web.springmvc.v1; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; @Controller public class SpringMemberFormControllerV1 { @RequestMapping("/springmvc/v1/members/new-form") public ModelAndView process() { System.out.println("SpringMemberFormControllerV1.process"); return new ModelAndView("new-form"); } }
main/java/hello.servlet/web/springmvc/v1/SpringMemberFormControllerV1 생성
- @Controller :
- 스프링이 자동으로 스프링 빈으로 등록한다. (내부에 @Component 애노테이션이 있어서 컴포넌트 스캔의 대상이 됨)
- 스프링 MVC에서 애노테이션 기반 컨트롤러로 인식한다.
- @RequestMapping : 요청 정보를 매핑한다. 해당 URL이 호출되면 이 메서드가 호출된다. 애노테이션을 기반으로 동작하기 때문에, 메서드의 이름은 임의로 지으면 된다.
- ModelAndView : 모델과 뷰 정보를 담아서 반환하면 된다.
실행
결과
RequestMappingHandlerMapping은 스프링 빈 중에서 @RequestMapping 또는 @Controller가 클래스 레벨에 붙어 있는 경우에 매핑 정보로 인식한다.
다음 코드도 동일하게 동작한다.
package hello.servlet.web.springmvc.v1; import org.springframework.stereotype.Component; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; //@Controller @Component @RequestMapping public class SpringMemberFormControllerV1 { @RequestMapping("/springmvc/v1/members/new-form") public ModelAndView process() { System.out.println("SpringMemberFormControllerV1.process"); return new ModelAndView("new-form"); } }
물론 컴포넌스 스캔 없이 다음과 같이 스프링 빈으로 직접 등록해도 동작한다.
@RequestMapping public class SpringMemberFormControllerV1 { @RequestMapping("/springmvc/v1/members/new-form") public ModelAndView process() { return new ModelAndView("new-form"); } }
주의! - 스프링 3.0 이상
스프링 부트 3.0(스프링 프레임워크 6.0)부터는 클래스 레벨에 @RequestMapping이 있어도 스프링 컨트롤러로 인식하지 않는다. 오직 @Controller가 있어야 스프링 컨트롤러로 인식한다. 참고로 @RestController는 해당 애노테이션 내부에 @Controller를 포함하고 있으므로 인식된다. 따라서 @Controller가 없는 위의 두 코드는 스프링 컨트롤러로 인식되지 않는다.
(RequestMappingHandlerMapping에서 @RequestMapping는 이제 인식하지 않고, Controller만 인식한다.)
SpringMemberSaveControllerV1 - 회원 저장
main/java/hello.servlet/web/springmvc/v1/SpringMemberSaveControllerV1 생성
package hello.servlet.web.springmvc.v1; import hello.servlet.domain.member.Member; import hello.servlet.domain.member.MemberRepository; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; @Controller public class SpringMemberSaveControllerV1 { private MemberRepository memberRepository = MemberRepository.getInstance(); @RequestMapping("/springmvc/v1/members/save") public ModelAndView process(HttpServletRequest request, HttpServletResponse response) { String username = request.getParameter("username"); int age = Integer.parseInt(request.getParameter("age")); Member member = new Member(username, age); memberRepository.save(member); ModelAndView mv = new ModelAndView("save-result"); mv.addObject("member", member); return mv; } }
- mv.addObject("member", member)
- 스프링이 제공하는 ModelAndView를 통해 Model 데이터를 추가할 때는 addObject()를 사용하면 된다. 이 데이터는 이후 뷰를 렌더링 할 때 사용된다.
SpringMemberListControllerV1 - 회원 목록
main/java/hello.servlet/web/springmvc/v1/SpringMemberListControllerV1 생성
package hello.servlet.web.springmvc.v1; import hello.servlet.domain.member.Member; import hello.servlet.domain.member.MemberRepository; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; import java.util.List; @Controller public class SpringmemberListControllerV1 { private final MemberRepository memberRepository = MemberRepository.getInstance(); @RequestMapping("/springmvc/v1/members") public ModelAndView process() { List<Member> members = memberRepository.findAll(); ModelAndView mv = new ModelAndView("members"); mv.addObject("members", members); return mv; } }
실행
스프링 MVC - 컨트롤러 통합
@RequestMapping을 잘 보면 클래스 단위가 아니라 메서드 단위에 적용된 것을 확인할 수 있다. 따라서 컨트롤러 클래스를 유연하게 하나로 통합할 수 있다.
SpringMemberControllerV2
main/java/hello.servlet/web/springmvc/v2/SpringMemberControllerV2 생성
package hello.servlet.web.springmvc.v2; import hello.servlet.domain.member.Member; import hello.servlet.domain.member.MemberRepository; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.util.List; /** * 클래스 단위 -> 메서드 단위 * @RequestMapping 클래스 레벨과 메서드 레벨 조합 */ @Controller public class SpringMemberControllerV2 { private MemberRepository memberRepository = MemberRepository.getInstance(); @RequestMapping("/springmvc/v2/members/new-form") public ModelAndView newForm() { System.out.println("SpringMemberControllerV2.newForm"); return new ModelAndView("new-form"); } @RequestMapping("/springmvc/v2/members/save") public ModelAndView save(HttpServletRequest request, HttpServletResponse response) { System.out.println("SpringMemberControllerV2.save"); String username = request.getParameter("username"); int age = Integer.parseInt(request.getParameter("age")); Member member = new Member(username, age); memberRepository.save(member); ModelAndView mv = new ModelAndView("save-result"); mv.addObject("member", member); return mv; } @RequestMapping("/springmvc/v2/members") public ModelAndView members() { System.out.println("SpringMemberControllerV2.members"); List<Member> members = memberRepository.findAll(); ModelAndView mv = new ModelAndView("members"); mv.addObject("members", members); return mv; } }
실행
결과
조합
컨트롤러 클래스를 통합하는 것을 넘어서 조합도 가능하다.
다음 코드는 /springmvc/v2/members라는 부분에 중복이 있다.
@RequestMapping("/springmvc/v2/members/new-form")
@RequestMapping("/springmvc/v2/members")
@RequestMapping("/springmvc/v2/members/save")
조합 결과
클래스 레벨 @RequestMapping("/springmvc/v2/members")
메서드 레벨 @RequestMapping("/new-form") -> /springmvc/v2/members/new-form
메서드 레벨 @RequestMapping("/save") -> /springmvc/v2/members/save
메서드 레벨 @RequestMapping -> /springmvc/v2/members
package hello.servlet.web.springmvc.v2; import hello.servlet.domain.member.Member; import hello.servlet.domain.member.MemberRepository; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.util.List; /** * 클래스 단위 -> 메서드 단위 * @RequestMapping 클래스 레벨과 메서드 레벨 조합 */ @Controller @RequestMapping("/springmvc/v2/members") public class SpringMemberControllerV2 { private MemberRepository memberRepository = MemberRepository.getInstance(); @RequestMapping("/new-form") public ModelAndView newForm() { System.out.println("SpringMemberControllerV2.newForm"); return new ModelAndView("new-form"); } @RequestMapping("/save") public ModelAndView save(HttpServletRequest request, HttpServletResponse response) { System.out.println("SpringMemberControllerV2.save"); String username = request.getParameter("username"); int age = Integer.parseInt(request.getParameter("age")); Member member = new Member(username, age); memberRepository.save(member); ModelAndView mv = new ModelAndView("save-result"); mv.addObject("member", member); return mv; } @RequestMapping public ModelAndView members() { System.out.println("SpringMemberControllerV2.members"); List<Member> members = memberRepository.findAll(); ModelAndView mv = new ModelAndView("members"); mv.addObject("members", members); return mv; } }
실행
스프링 MVC - 실용적인 방식
MVC 프레임워크 만들기에서 v3은 ModelView를 개발자가 직접 생성해서 반환했기 때문에, 불편했던 기억이 날 것이다. 물론 v4를 만들면서 실용적으로 개선한 기억도 날 것이다.
스프링 MVC는 개발자가 편리하게 개발할 수 있도록 수많은 편의 기능을 제공한다.
실무에서는 지금부터 설명하는 방식을 주로 사용한다.
SpringMemberControllerV3
main/java/hello.servlet/web/springmvc/v3/SpringMemberControllerV3 생성
package hello.servlet.web.springmvc.v3; import hello.servlet.domain.member.Member; import hello.servlet.domain.member.MemberRepository; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestParam; import java.util.List; /** * Model 도입 * ViewName 직접 변환 * @RequestParam 사용 */ @Controller @RequestMapping("/springmvc/v3/members") public class SpringMemberControllerV3 { private MemberRepository memberRepository = MemberRepository.getInstance(); // 애노테이션 기반 컨트롤러는 ModelAndView를 반환하는것이 가능하고 문자 반환도 가능 @RequestMapping("/new-form") public String newForm() { System.out.println("SpringMemberControllerV3.newForm"); return "new-form"; } // request, response를 받을 수 있지만 파라미터를 직접 받을 수 있음 @RequestMapping("/save") public String save(@RequestParam("username") String username, @RequestParam("age") int age, Model model) { System.out.println("SpringMemberControllerV3.save"); Member member = new Member(username, age); memberRepository.save(member); model.addAttribute("member", member); return "save-result"; } @RequestMapping public String members(Model model) { System.out.println("SpringMemberControllerV3.members"); 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 방식을 모두 지원한다.
실행
GET, POST 구분
지금까지는 GET, POST 상관없이 실행 가능
// 애노테이션 기반 컨트롤러는 ModelAndView를 반환하는것이 가능하고 문자 반환도 가능 @RequestMapping(value = "/new-form", method = RequestMethod.GET) public String newForm() { System.out.println("SpringMemberControllerV3.newForm"); return "new-form"; }
method = RequestMethod.GET 추가
GET은 가능하지만 POST는 불가능
package hello.servlet.web.springmvc.v3; import hello.servlet.domain.member.Member; import hello.servlet.domain.member.MemberRepository; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.*; import java.util.List; /** * Model 도입 * ViewName 직접 변환 * @RequestParam 사용 */ @Controller @RequestMapping("/springmvc/v3/members") public class SpringMemberControllerV3 { private MemberRepository memberRepository = MemberRepository.getInstance(); // 애노테이션 기반 컨트롤러는 ModelAndView를 반환하는것이 가능하고 문자 반환도 가능 @GetMapping("/new-form") public String newForm() { System.out.println("SpringMemberControllerV3.newForm"); return "new-form"; } // request, response를 받을 수 있지만 파라미터를 직접 받을 수 있음 @PostMapping("/save") public String save(@RequestParam("username") String username, @RequestParam("age") int age, Model model) { System.out.println("SpringMemberControllerV3.save"); Member member = new Member(username, age); memberRepository.save(member); model.addAttribute("member", member); return "save-result"; } @GetMapping public String members(Model model) { System.out.println("SpringMemberControllerV3.members"); List<Member> members = memberRepository.findAll(); model.addAttribute("members", members); return "members"; } }
@RequestMapping -> @GetMapping, @PostMapping
@RequestMapping 은 URL만 매칭하는 것이 아니라, HTTP Method도 함께 구분할 수 있다.
예를 들어서 URL이 /new-form이고, HTTP Method가 GET인 경우를 모두 만족하는 매핑을 하려면 다음과 같이 처리하면 된다.
@RequestMapping(value = "/new-form", method = RequestMethod.GET)
이것을 @GetMapping, @PostMapping으로 더 편리하게 사용할 수 있다.
참고로 Get, Post, Put, Delete, Patch 모두 애노테이션이 준비되어 있다.
출처 : https://www.inflearn.com/course/%EC%8A%A4%ED%94%84%EB%A7%81-mvc-1
'Spring > 스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술' 카테고리의 다른 글
[김영한 스프링] 21. 스프링 MVC 기본 기능 - 로깅 간단히 알아보기 (0) 2023.07.11 [김영한 스프링] 20. 스프링 MVC 기본 기능 - 프로젝트 생성 & 세팅 (0) 2023.07.06 [김영한 스프링] 18. 스프링 MVC 구조 이해 - 전체 구조 & 핸들러 매핑과 핸들러 어댑터 & 뷰 리졸버 (0) 2023.07.04 [김영한 스프링] 17. MVC 프레임워크 만들기 - 유연한 컨트롤러 (0) 2023.07.01 [김영한 스프링] 16. MVC 프레임워크 만들기 - 단순하고 실용적인 컨트롤러 (0) 2023.06.30