ABOUT ME

Today
Yesterday
Total
  • [스프링MVC - 1편] 스프링 MVC 기본기능 - 3
    스프링&스프링부트 2025. 1. 29. 12:25

    HTTP 응답 - 정적 리소스, 템플릿

    스프링(서버)에서 응답 데이터를 만드는 방법 3가지

    • 정적 리소스    - 예)  브라우저에 정적인 HTML, css, js 제공할 때는 정적 리소스를 사용한다
    • 뷰 템플릿 사용  - 예)  브라우저에 동적인 HTML 제공할 때는  템플릿을 사용한다
    • HTTP 메시지 사용 - 예) HTTP API를 제공하는 경우에는 HTML 아니라 데이터를 전달해야 하므로, HTTP 메시지 바디에 JSON 같은 형식으로 데이터를 실어 보낸다

     

     

    1. 정적 리소스

    스프링 부트는 클래스패스의 다음 디렉토리에 있는 정적 리소스를 제공한다

    /static, /public, /resources, /META-INF/resources

     

    src/main/resources 리소스를 보관하는 곳이고, 클래스패스의 시작 경로이다

    따라서 다음 디렉토리에 리소스를 넣어두면 스프링 부트가 정적 리소스로 서비스를 제공한다

    정적 리소스 경로 - src/main/resources/static

     

    src/main/resources/static/basic/hello-form.html 경로에 파일이 들어있으면

    브라우저에서 http://localhost:8080/basic/hello-form.html 실행하면 된다

    정적 리소스는 해당 파일을 변경 없이 그대로 서비스하는 것이다

     

     

    2. 뷰 템플릿

    템플릿을 거쳐서 HTML 생성되고, 뷰가 응답을 만들어서 전달한다

    일반적으로 HTML 동적으로 생성하는 용도로 사용하지만, 다른 것들도 가능하다

     

    스프링 부트는 기본 템플릿 경로를 제공한다

    템플릿 경로 - src/main/resources/templates

     

     

    2-1. 템플릿 생성 - hello.html

     

     

    2-2. ResponseViewController - 템플릿을 호출하는 컨트롤러

    @Controller
    public class ResponseViewController {
        @RequestMapping("/response-view-v1")
        public ModelAndView responseViewV1() {
            ModelAndView mav = new ModelAndView("response/hello").addObject("data", "hello!");
            return mav;
        }
    }

     

     

     

    String 반환하는 경우 - View or HTTP 메시지

    @RequestMapping("/response-view-v2")
    public String responseViewV2(Model model) {
        model.addAttribute("data", "hello!");
        return "response/hello";
    }

    @ResponseBody 없으면 response/hello로 리졸버가 실행되어서 뷰를 찾고, 렌더링 한다

    @ResponseBody 있으면 리졸버를 실행하지 않고, HTTP 메시지 바디에 직접 response/hello라는 문자가 입력된다

    여기서는 뷰의 논리 이름인 response/hello 반환하면 다음 경로의 템플릿이 렌더링 되는 것을 확인할

    > 실행: templates/response/hello.html

     

     

    Void 반환하는 경우

    @RequestMapping("/response/hello")
    public void responseViewV3(Model model) {
        model.addAttribute("data", "hello!");
    }

    @Controller 사용하고, HttpServletResponse, OutputStream(Writer)같은 HTTP 메시지 디를 처리하는 파라미터가 없으면 요청 URL 참고해서 논리 이름으로 사용한다

    > 요청 URL: /response/hello

    > 실행: templates/response/hello.html

    참고로 방식은 명시성이 너무 떨어지고 이렇게 맞는 경우도 많이 없어서, 권장하지 않는다

     

    HTTP 메시지

    @ResponseBody, HttpEntity 사용하면, 템플릿을 사용하는 것이 아니라, HTTP 메시지 바디에 직접 응답데이터를 출력할 있다

     

     

     

    HTTP 응답 - HTTP API, 메시지 바디에 직접 입력

    HTTP API 제공하는 경우에는 HTML 아니라 데이터를 전달해야 하므로, HTTP 메시지 바디에 JSON 같은 형식으로 데이터를 실어 보낸다

     

     

    ResponseBodyController

    responseBodyV1

    @GetMapping("response-body-string-v1")
    public void responseBodyV1(HttpServletResponse response) throws IOException {
        response.getWriter().write("ok");
    }

    서블릿을 직접 다룰 때 처럼HttpServletResponse 객체를 통해서 HTTP 메시지 바디에 직접 ok 응답 메시지를 전달한다

    response.getWriter().write("ok")

     

     

    responseBodyV2

    @GetMapping("response-body-string-v2")
    public ResponseEntity<String> responseBodyV2(HttpServletResponse response) {
        return new ResponseEntity<>("ok", HttpStatus.OK);
    }

    ResponseEntity 엔티티는 HttpEntity를 상속받았는데, HttpEntity HTTP 메시지의 헤더, 바디 정보를 지고 있다

    ResponseEntity 여기에 더해서 HTTP 응답 코드를 설정할 있다

    HttpStatus.CREATED로 변경하면 201 응답이 나가는 것을 확인할 있다

     

     

    responseBodyV3

    @ResponseBody
    @GetMapping("response-body-string-v3")
    public String responseBodyV3() {
        return "ok";
    }

    @ResponseBody 사용하면 view 사용하지 않고, HTTP 메시지 컨버터를 통해서 HTTP 메시지를 직접입력할 수

    ResponseEntity도 동일한 방식으로 동작한다

     

     

    responseBodyJsonV1

    @GetMapping("response-body-json-v1")
    public ResponseEntity<HelloData> responseBodyJsonV1(){
        HelloData helloData = new HelloData();
        helloData.setUsername("userA");
        helloData.setAge(20);
        return new ResponseEntity<>(helloData, HttpStatus.OK);
    }

    ResponseEntity 반환한다

    HTTP 메시지 컨버터를 통해서 JSON 형식으로 변환되어서 반환된다

     

     

    responseBodyJsonV2

    @ResponseStatus(HttpStatus.OK)
    @ResponseBody
    @GetMapping("response-body-json-v2")
    public HelloData responseBodyJsonV2(){
        HelloData helloData = new HelloData();
        helloData.setUsername("userA");
        helloData.setAge(20);
        return helloData;
    }

    ResponseEntity HTTP 응답 코드를 설정할 있는데, @ResponseBody 사용하면 이런 것을 설정하기 까다롭다

    @ResponseStatus(HttpStatus.OK) 애노테이션을 사용하면 응답 코드도 설정할 있다

    물론 애노테이션이기 때문에 응답 코드를 동적으로 변경할 수는 없다

    프로그램 조건에 따라서 동적으로 변경하려면 ResponseEntity 사용하면 된다

     

     

    @RestController

    @Controller 대신에 @RestController 애노테이션을 사용하면, 해당 컨트롤러에 모두 @ResponseBody적용되는 효과가 있다

    따라서 템플릿을 사용하는 것이 아니라, HTTP 메시지 바디에 직접 데이터를 입력한다

    이름 그대로 Rest API(HTTP API) 만들 사용하는 컨트롤러이다

    참고로 @ResponseBody 클래스 레벨에 두면 전체 메서드에 적용되는데, @RestController 에노테이션 안에 @ResponseBody 적용되어 있다

    728x90
Designed by Tistory.