ABOUT ME

-

Today
-
Yesterday
-
Total
-
  • [김영한 스프링] 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

     

    스프링 MVC 1편 - 백엔드 웹 개발 핵심 기술 - 인프런 | 강의

    웹 애플리케이션을 개발할 때 필요한 모든 웹 기술을 기초부터 이해하고, 완성할 수 있습니다. 스프링 MVC의 핵심 원리와 구조를 이해하고, 더 깊이있는 백엔드 개발자로 성장할 수 있습니다., 원

    www.inflearn.com

Designed by Tistory.