프로젝트 선택
Project Metadata
Dependencies: Spring Web, Thymeleaf, Lombok
주의!
Packaging는 War가 아니라 Jar를 선택한다. 톰캣과 같은 was서버를 별도로 설치하고 거기에 빌드된 파일을 넣을 때, 또는 JSP를 쓸 때 WAR를 사용한다. 그 외 내장톰캣에 최적화해서 바로 사용하고 싶을때는 JAR를 사용한다. Jar를 사용하면 항상 내장 서버(톰캣등)를 사용하고.webapp
경로도 사용하지 않는다. 내장 서버 사용에 최적화되어 있는 기능이므로 주로 이방식을 사용한다. War를 사용하면 내장 서버도 사용 가능하지만, 주로 외부 서버에 배포한는 목적으로 사용한다.
동작확인
기본메인클래스 실행( SpringmvcApplication.main() )
http://localhost:8080 호출해서 Whitelabel Error Page가나오면 정상동작
스프링 부트에 Jar
를 사용하면 /resources/static
위치에 index.html
파일을 두면 Welcome 페이지로 처리해준다.(스트링 부트가 지원하는 정적 컨텐츠 위에 `/index.html
이 있으면 된다.
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<ul>
<li>로그 출력
<ul>
<li><a href="/log-test">로그 테스트</a></li>
</ul>
</li>
<!-- -->
<li>요청 매핑
<ul>
<li><a href="/hello-basic">hello-basic</a></li><li><a href="/mapping-get-v1">HTTP 메서드 매핑</a></li>
<li><a href="/mapping-get-v2">HTTP 메서드 매핑 축약</a></li>
<li><a href="/mapping/userA">경로 변수</a></li>
<li><a href="/mapping/users/userA/orders/100">경로 변수 다중</a></li>
<li><a href="/mapping-param?mode=debug">특정 파라미터 조건 매핑</a></li>
<li><a href="/mapping-header">특정 헤더 조건 매핑(POST MAN 필요)</a></
<li><a href="/mapping-consume">미디어 타입 조건 매핑 Content-Type(POST
li>
MAN 필요)</a></li>
<li><a href="/mapping-produce">미디어 타입 조건 매핑 Accept(POST MAN
필요)</a></li>
</ul>
</li>
<li>요청 매핑 - API 예시
<ul>
<li>POST MAN 필요</li>
</ul>
</li>
<li>HTTP 요청 기본
<ul>
<li><a href="/headers">기본, 헤더 조회</a></li>
</ul>
</li>
<li>HTTP 요청 파라미터
<ul>
<li><a href="/request-param-v1?username=hello&age=20">요청 파라미터
v1</a></li>
<li><a href="/request-param-v2?username=hello&age=20">요청 파라미터
v2</a></li>
<li><a href="/request-param-v3?username=hello&age=20">요청 파라미터
v3</a></li>
<li><a href="/request-param-v4?username=hello&age=20">요청 파라미터
v4</a></li>
<li><a href<li><a href="/model-attribute-v1?username=hello&age=20">요청 파라미터
@ModelAttribute v1</a></li>
<li><a href="/model-attribute-v2?username=hello&age=20">요청 파라미터
@ModelAttribute v2</a></li>
</ul>
</li>
<li>HTTP 요청 메시지
<ul>
<li>POST MAN</li>
</ul>
</li>
<li>HTTP 응답 - 정적 리소스, 뷰 템플릿
<ul>
<li><a href="/basic/hello-form.html">정적 리소스</a></li>
<li><a href="/response-view-v1">뷰 템플릿 v1</a></li>
<li><a href="/response-view-v2">뷰 템플릿 v2</a></li>
</ul>
</li>
<li>HTTP 응답 - HTTP API, 메시지 바디에 직접 입력
<ul>
<li><a href="/response-body-string-v1">HTTP API String v1</a></li>
<li><a href="/response-body-string-v2">HTTP API String v2</a></li>
<li><a href="/response-body-string-v3">HTTP API String v3</a></li>
<li><a href="/response-body-json-v1">HTTP API Json v1</a></li>
<li><a href="/response-body-json-v2">HTTP API Json v2</a></li>
</ul>
</li>
</ul>
</body>
</html>
앞으로 로그를 사용할 것이기 때문에, 이번 시간에는 로그에 대해서 간단히 알아보자.
운영시스템에서는 System.out.println()
같은 시스템 콘솔을 사용해서 필요한 정보를 출력하지 않고, 별도의 로깅 라이브러리를 사용해서 로그를 출력한다. 참고로 로그 관련 라이브러리도 많고, 깊게 들어가면 끝이 없기 때문에, 여기서는 최소한의 사용 방법만 알아본다.
스프링 부트 라이브러리를 사용하면 스프링 부트 로깅 라이브러리(spring-boot-starter-logging
)가 함께 포함된다. 스프링 부트 로깅 라이브러리는 기본으로 다음 로깅 라이브러리를 사용한다.
3가지 방법 모두 사용 가능.
private Logger log = LoggerFactory.getLogger(getClass());
private static final Logger log = LoggerFactory.getLogger(Xxx.class)
@Slf4j
: 롬복사용가능 (애노테이션만 붙여줘도 사용 가능)
log.info("hello")
System.out.println("hello")
package hello.springmvc.basic;
import lombok.extern.slf4j.Slf4j;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
//@Slf4j
@RestController
public class LogTestController {
private final Logger log = LoggerFactory.getLogger(getClass());
@RequestMapping("/log-test")
public String logTest() {
String name = "Spring";
System.out.println("name = " + name);
log.trace("trace log={}", name);
log.debug("debug log={}", name);
log.info("info log={}", name);
log.warn("warn log={}", name);
log.error("error log={}", name);
// 로그를 사용하지 않아도 a+b 계산 로직이 먼저 실행됨, 이런 방식으로 사용하면 x
log.debug("String concat log = " + name);
return "ok";
}
}
@RestController
@Controller
는 반환값이 String이면 뷰 이름으로 인식한다. 그래서 뷰를 찾고 뷰가 렌더링 된다.@RestController
는 반환값이 String이면 뷰를 찾는게 아니라, HTTP 메시지 바디에 바로 입력한다. 따라서 실행결과로 ok메시지를 받을 수 있다. @ResponseBody
와 관련이 있는데, 뒤에서 더 자세히 설명한다.시간, 로그레벨(INFO), 프로세스ID, [현재 실행한 쓰레드명], 현재나의 컨트롤러 이름,: 로그 메세지
로그레벨 설정을 변경해서 출력 결과를 보자
TRACE > DEBUG > INFO > WARN > ERROR
로그를 찍을 때 레벨을 정할 수 있음.
ex)
debug: 현재로그는 디버그할 때, 개발 서버에서 보는 거야
info: 현재 로그는 중요한 비즈니스 정보야. 운영시스템에서도 봐야 할 정보야
warn: 이거는 위험한거야.
error: 이거는 에러야. 우리가 빨리 확인해야해. error가 뜨면 알람을 보거나 별도로 어디 파일에 남겨서 보거나 등등 할 수 있다.
application.properties
#전체 로그 레벨 설정(기본 info)
logging.level.root=info
#hello.springmvc 패키지와 그 하위 로그 레벨 설정
logging.level.hello.springmvc=debug
로그 레벨에 맞춰 출력되는 로그를 조절 -> 중요한 정보만 선택적으로 출력 가능. 트래픽이 많아 출력되는 로그가 과도하게 많을 때, 필요한 정보만 선택적으로 출력해 볼 수 있다는 장점!
기본셋팅: info부터 출력
(if. 이걸 debug로 바꾸면.. 모든 라이브러리에 다 log가 있음. 스프링부트가 어마어마한 정보들을 출력하게 됨. 기본은 info로 두고 원하는 부분만 살짝식 바꾸는게 좋다.)
log.debug("data="+data)
log.debug("data={}", data);
로그에대해서 더 자세한내용은 slf4j, logback을검색해보자.
요청매핑 : 요청이 왔을 때 어떤 컨트롤러가 호출되어야 하는지 매핑 하는 것 (단순하게 url을 통해 매핑하는 방법 뿐만 아니라, 여러 요소를 조합해서 매핑한다.)
package hello.springmvc.basic.requestmapping;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.annotation.*;
@RestController
public class MappingController {
private Logger log = LoggerFactory.getLogger(getClass());
/**
* 기본 요청
* 둘다 허용 /hello-basic, /hello-basic/
* HTTP 메서드 모두 허용 GET, HEAD, POST, PUT, PATCH, DELETE
*/
@RequestMapping("/hello-basic")
public String helloBasic() {
log.info("helloBasic");
return "ok";
}
}
매핑 정보(한번 더)
@RestController
@Controller
는 반환값이 String이면 뷰 이름으로 인식된다. 그래서 뷰를 찾고 뷰가 랜더링된다.@RestController
는 반환값으로 뷰를 찾는 것이 아니라, HTTP 메시지 바디에 바로 입력한다. 따라서 실행 결과로 ok메시지를 받을 수 있다. @ResponseBody
와 관련이 있는데, 뒤에서 더 자세히 설명한다.@RequestMapping("/hello-basic")
/hello-basic
URL 호출이 오면 이 메서드가 실행되도록 매핑한다.배열[]
로 제공하므로 다중 설정이 가능하다. {“/hello-baskc", "/hello-go"}
둘 다 허용 - 스프링 부트 3.0 이전
다음 두가지 요청은 다른 URL이지만, 스프링은 다음 URL 요청들을 같은 요청으로 매핑한다.
- 매핑:
/hello-basic
- URL 요청:
/hello-basic
,/hello-basic/
스프링부트 3.0 이후
스프링부트 3.0 부터는/hello-basic
,/hello-basic/
는 서로 다른 URL 요청을 사용해야 한다. 기존에는 마지막에 있는/
(slash)를 제거했지만, 스프링부트 3.0 부터는 마지막의/
(slash)를 유지한다. 따라서 다음과 같이 다르게 매핑해서 사용해야 한다.
- 매핑:
/hello-basic
=> URL 요청:/hello-basic
- 매핑:
/hello-basic/
=> URL 요청:/hello-basic/
@RequestMapping
에 method
속성으로 HTTP 메서드를 지정하지 않으면 HTTP 메서드와 무관하게 호출된다. 모두 허용 GET, HEAD, POST, PUT, PATCH, DELETE
HTTP 메서드 매핑
/**
* method 특정 HTTP 메서드 요청만 허용 * GET, HEAD, POST, PUT, PATCH, DELETE
*/
@RequestMapping(value = "/mapping-get-v1", method = RequestMethod.GET)
public String mappingGetV1() {
log.info("mappingGetV1");
return "ok";
}
만약 여기에 POST 요청을 하면 스프링 MVC는 HTTP 405상태코드(Method Not Allowed)를 반환한다.
HTTP 메서드 매핑 축약
/**
* 편리한 축약 애노테이션 (코드보기)
* @GetMapping
* @PostMapping
* @PutMapping
* @DeleteMapping
* @PatchMapping
*/
@GetMapping(value = "/mapping-get-v2")
public String mappingGetV2() {
log.info("mapping-get-v2");
return "ok";
}
HTTP 메서드를 축약한 애노테이션을 사용하는 것이 더 직관적이다. 코드를 보면 내부에서 @RequestMapping
과 method
를 지정해서 사용하는 것을 확인할 수 있다.
요즘엔 이런 경로 스타일을 사용 많이들 한다!
/**
* PathVariable 사용
* 변수명이 같으면 생략 가능
* @PathVariable("userId") String userId -> @PathVariable userId
* /mapping/userA (URL자체에 값이 들어있는 것.)
*/
@GetMapping("/mapping/{userId}")
public String mappingPath(@PathVariable("userId") String data) {
log.info("mappingPath userId={}", data);
return "ok";
}
최근 HTTP API는 다음과 같이 리소스 경로에 식별자를 넣는 스타일을 선호한다.
/mapping/userA
/users/1
@RequestMapping
은 URL 경로를 템플릿화 할 수 있는데, @PathVariable
를 사용하면 매칭 되는 부분을 편리하게 조회할 수 있다.
템플릿화: 임의의 값이 들어오도록 설정하는 것. 여기서는
{userId}
부분
@PathVariable
의 이름과 파라미터 이름이 같으면 생략할 수 있다.
변수명을 통일하면 불러오는 컬럼명을 지우고 사용할 수 있다.
@GetMapping("/mapping/{userId}")
public String mappingPath(@PathVariable String userId) {
log.info("mappingPath userId={}", userId);
return "ok";
}
cf. 쿼리 파라미터 형식:
maping?userId=userA
경로 변수 형식:/mapping/userA
PathVariable 사용 - 다중
/**
* PathVariable 사용 다중
*/
@GetMapping("/mapping/users/{userId}/orders/{orderId}")
public String mappingPath(@PathVariable String userId, @PathVariable Long orderId) {
log.info("mappingPath userId={}, orderId={}", userId, orderId);
return "ok";
}
실행
http://localhost:8080/mapping/users/userA/orders/100
users=userA, orders=100
/**
* 파라미터로 추가 매핑
* params="mode",
* params="!mode"
* params="mode=debug"
* params="mode!=debug"
* params={"mode=debug","data=good"}
*/
@GetMapping(value="/mapping-param", params = "mode=debug")
public String mappingParam() {
log.info("mappingParam");
return "ok";
}
잘 사용하지는 않는다. 특정 파라미터가 있거나 없는 조건을 추가할 수 있다.
(특정 파라미터가 같이 넘어와야지만 해당 매핑이 실행됨.)
실행
http://localhost:8080/mapping-param?mode=debug
/**
* 특정 헤더로 추가 매핑
* headers="mode",
* headers="!mode"
* headers="mode=debug"
* headers="mode!=debug"
*/
@GetMapping(value="/mapping-header", headers="mode=debug")
public String mappingHeader() {
log.info("mappingHeader");
return "ok";
}
파라미터 매핑과 비슷하지만, HTTP 헤더를 사용한다.
/**
* Content-Type 헤더 기반 추가 매핑 Media Type
* consumes="application/json"
* consumes="!application/json"
* consumes="application/*"
* consumes="*\/*"
* MediaType.APPLICATION_JSON_VALUE
*/
@PostMapping(value = "/mapping-consume", consumes = "application/json")
public String mappingConsumes() {
log.info("mappingConsumes");
return "ok";
}
HTTP 요청의 Content-Type 헤더를 기반으로 미디어 타입으로 매핑한다. 만약 맞지 않으면 HTTP 415 상태코드(Unsupported Media Type)을 반환한다.
예시)consumes
consumes = "text/plain"
consumes = {"text/plain", "application/*"}
consumes = MediaType.TEXT_PLAIN_VALUE
consume: 내가 소비하는 입장. 내가 해당 정보를 소비하고 있기 때문에 consume.
/**
* Accept 헤더 기반 Media Type
* produces = "text/html"
* produces = "!text/html" * produces = "text/*"
* produces = "*\/*"
*/
@PostMapping(value = "/mapping-produce", produces = "text/html")
public String mappingProduces() {
log.info("mappingProduces");
return "ok";
}
HTTP 요청의 Accetp 헤더를 기반으로 미디어 타입으로 매핑한다. 만약 맞지 않으면 HTTP 406 상태코드(Not Acceptable)을 반환한다.
producers = "text/html"
: 클라이언트가 나는 text:html형식을 받아들일 수 있어, 하는 의미!
예시)
produces = "text/plain"
produces = {"text/plain", "application/*"}
produces = MediaType.TEXT_PLAIN_VALUE
produces = "text/plain;charset=UTF-8"
Content-Type 헤더 vs Accept 헤더
- Content-Type 헤더
: HTTP 메시지(요청과 응답 모두)에 담겨 보내는 데이터의 형식을 알려주는 헤더. 대부분 Content-Type헤더를 기준으로 HTTP 메시지에 담긴 데이터를 파싱한다.
: HTTP 요청의 GET방식인 경우, 무조건 URL 끝에 쿼리스트링 형식으로 날아오기때문에 Content-Type의 헤더가 필요없다.(웹 서버에서도 method가 GET이면 key=value형식임을 유추할 수 있기 때문)
=> 따라서 Content-Type은 POST나 PUT처럼 메시지 바디에 데이터를 심어보낼때 중요하다.- Accept 헤더
: 브라우저(클라이언트)에서 웹 서버로 요청시 요청메시지에 담기는 헤더.
: "자신이 이러한 데이터 타입만 허용하겠다는 뜻"
: ex.Accept헤더:application/json
= "나는 json 데이터만 처리할 수 있으니 json 데이터 형식으로 응답을 돌려주세요."- 결론: 둘다 데이터타입을 다루는 헤더이지만 Content-Type헤더는 현재 전송하는 데이터가 어떤 타입인지에 대한 설명이고 Accept해더는 클라이언트가 서버에게 웬만하면 데이터 전송할때 이러이러한 타입만 가공해서 보내라 라고 하는 것과 같다.
출처:https://dololak.tistory.com/630
회원 관리를 HTTP API로 만든다 생각하고 매핑을 어떻게 하는지 알아보자.
(실제 데이터가 넘어가는 부분은 생략하고 URL 매핑만)
/users
/users
/users/{uesrId}
/users/{userId}
/users/{userId}
package hello.springmvc.basic.requestmapping;
import org.springframework.web.bind.annotation.*;
@RestController
@RequestMapping("/mapping/users")
public class MappingClassController {
/**
* < 회원관리 API >
* 회원목록조회: GET /users
* 회원등록: POST /users
* 회원조회: GET /users/{userId}
* 회원수정: PATCH /users/{userId}
* 회원삭제: DELETE /users/{userId}
*/
@GetMapping
public String user() {
return "get users";
}
@PostMapping
public String addUser(){
return "post user";
}
@GetMapping("/{userId}")
public String findUser(@PathVariable String userId) {
return "get userId = " + userId;
}
@PatchMapping("/{userId}")
public String updateUser(@PathVariable String userId) {
return "update userId="+ userId;
}
@DeleteMapping("/{userId}")
public String deleteUser(@PathVariable String userId) {
return "delete userId="+userId;
}
}
/mappin
: 강의의 다른 예제들과 구분하기 위해 사용@RequestMapping("/mapping/users")
: 클래스 레벨에 매핑정보를 두면 메서드 레벨에서 해당 정보를 조합해서 사용한다.Postman으로 테스트
애노테이션 기반의 스프링 컨트롤러는 다양한 파라미터를 지원한다. 이번 시간에는 HTTP 헤더 정보를 조회하는 방법을 알아보자.
애노테이션 기반의 스프리 컨트롤러: RequestMappingHandlerAdapter
앞서 2장 서블릿에서도 HttpServletRequest를 통해 헤더정보를 불러오는 것을 연습했다. 스프링이 지원하는 것들을 통해 해당 정보들을 간단히 불러 올 수 있다. 이를 확인해보자!
package hello.springmvc.basic.request;
import lombok.extern.slf4j.Slf4j;
import org.springframework.http.HttpMethod;
import org.springframework.util.MultiValueMap;
import org.springframework.web.bind.annotation.CookieValue;
import org.springframework.web.bind.annotation.RequestHeader;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Locale;
@Slf4j
@RestController
public class RequestHeaderController {
@RequestMapping("/headers")
public String headers(HttpServletRequest request,
HttpServletResponse response,
HttpMethod httpMethod,
Locale locale,
@RequestHeader MultiValueMap<String, String> headerMap,
@RequestHeader("host") String host,
@CookieValue(value="myCookie", required = false) String cookie
) {
// 애노테이션 기반의 컨트롤러는 정말 다양한 파라미터를 받아들일 수 있다.
// 인터페이스로 정형화 되어 있는것이 아니기 때문에, 스프링이 지원하는 것은 다 가능하다.
log.info("request={}", request);
log.info("response={}", response);
log.info("httpMethod={}", httpMethod);
log.info("locale={}", locale);
log.info("headerMap={}", headerMap);
log.info("header host={}", host);
log.info("myCookie={}", cookie);
return "ok";
}
}
HttpServletRequest
HttpServletResponse
HttpMethod
: HTTP 메서드를 조회한다. org.springframework.http.HttpMethod
Locale
: Locale 정보를 조회한다.
스프링에는 여러가지 locale이 왔을때 어느 locale을 가장 우선으로 둘지 지정하는 localeResolver가 있음. locale을 쿠키나 세션에 두고 저장하면서서 사용하는 것이 가능. 기본은 http가 보내는 것을 사용하는 데 그것을 넘어 다른 방식으로 사용할 수도 있다. 더 알고싶으면 localeResolver에 대해 찾아보자!
@RequestHeader MultiValueMap<String, String> headerMap
: 모든 HTTP 헤더를 MultiValueMap 형식으로 조회한다.
@RequestHeader("hose") String host
required
/ 기본값 속성 = defaultValue
MultiValueMap
Map과 유사한데 하나의 키에 여러 값을 받을 수 있다.
HTTP header, HTTP 쿼리 파라미터와 같이 하나의 키에 여러 값을 받을 때 사용한다.
keyA=value1&vkeyA=value2
MultiValueMap<String, String> map = new LinkedMultiValueMap();
map.add("keyA", "value1");
map.add("keyA", "value2");
//[value1,value2]
List<String> values = map.get("keyA");
같은 헤더에 같은 값들이 들어올 수 있음. 그런 경우 MultiValueMap으로 받으면 된다. keyA를 꺼내면 배열이 반환되어 해당 key의 값들이 다 들어있다.(value1,value2)
@Slf4j
log
라고 사용하면 된다.private static final org.slf4j.Logger log =
org.slf4j.LoggerFactory.getLogger(RequestHeaderController.class);
참고
@Conroller
의 사용 가능한 파라미터 목록은 다음 공식메뉴얼에서 확인할 수 있다.
https://docs.spring.io/spring-framework/docs/current/reference/html/web.html#mvc-ann-
arguments
@Conroller
의 사용 가능한 응답값 목록은 다음 공식메뉴얼에서 확인할 수 있다.
https://docs.spring.io/spring-framework/docs/current/reference/html/web.html#mvc-ann-return-types
각각의 애노테이션에 대한 메뉴얼 링크도 연결되어 있음.
서블릿에서의 헤더 정보 요청
private void printHeaderUtils(HttpServletRequest request) { System.out.println("--- Header 편의 조회 start ---"); System.out.println("[Host 편의 조회]"); System.out.println("request.getServerName() = " + request.getServerName()); //Host 헤더 System.out.println("request.getServerPort() = " + request.getServerPort()); //Host 헤더 System.out.println(); System.out.println("[Accept-Language 편의 조회]"); request.getLocales().asIterator() .forEachRemaining(locale -> System.out.println("locale = " + locale)); System.out.println("request.getLocale() = " + request.getLocale()); System.out.println(); System.out.println("[cookie 편의 조회]"); if (request.getCookies() != null) { for (Cookie cookie : request.getCookies()) { System.out.println(cookie.getName() + ": " + cookie.getValue()); } } System.out.println(); System.out.println("[Content 편의 조회]"); System.out.println("request.getContentType() = " + request.getContentType()); System.out.println("request.getContentLength() = " + request.getContentLength()); System.out.println("request.getCharacterEncoding() = " + request.getCharacterEncoding()); System.out.println("--- Header 편의 조회 end ---"); System.out.println(); }
엄청 복잡했다. 스프링에서 제공하는 기능덕에 매우 편리해졌음을 확인 가능하다.
서블릿에서 학습했던 HTTP 요청 데이터를 조회하는 방법을 다시 떠올려보자. 그릐고 서블릿으로 학습했던 내용을 스프링이 얼마나 깔끔하고 효율적으로 바꾸어주는지 알아보자.
HTTP 요청 메시지를 통해 클라이언트에서 서버로 데이터를 전달하는 방법은 3가지가 있다.
/url?username=hello&age=20
content-type:application/x-www-form-urlencoded
username=hello&age=20
HttpServletRequest
의 request.getParameter()
를 사용하면 다음 두 가지 요청 파라미터를 조회할 수 있다.
GET, 쿼리 파라미터 전송
예시:http://localhost:8080/request-param?username=hello&age=20
POST, HTML Form 전송
예시:
POST /request-param ...
content-type: application/x-www-form-urlencoded
username=hello&age=20
GET 쿼리 파라미터 전송 방식이든 POST HTML Form 전송 방식이든 둘다 형식이 같으므로 구분 없이 조회할 수 있다. 이것을 간단히 요청 파라미터(request parameter) 조회라 한다.
지금부터 스프링으로 요청 파라미터를 조회하는 방법을 단계적으로 알아보자.
이번 단락에선, request.getParameter("변수명")를 통해 get, post방식의 데이터 조회를 확인한다.
RequestParamController
package hello.springmvc.basic.request;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@Slf4j
@Controller
public class RequestParamController {
/*
반환 타입이 없으면서 이렇게 응답에 값을 직접 집어넣으면, view 조회X
*/
@RequestMapping("/request-param-v1")
public void requestParamV1(HttpServletRequest request, HttpServletResponse response) throws IOException {
String username = request.getParameter("username");
int age = Integer.parseInt(request.getParameter("age"));
log.info("username={}, age={}", username, age);
response.getWriter().write("ok");
}
}
여기서는 단순히 HttpServletRequest가 제공하는 방식으로 요청 파라미터를 조회햇다.
http://localhost:8080/request-param-v1?username=hello&age=20
먼저 테스트용 HTML Form을 만들어야 한다.
리소스는 /resources/static
아래에 두면 스프링 부트가 자동으로 인식한다.
resurces의 static 안의 파일들은 외부에 공개된다.(그래서 index.html 파일도 공개가 된것)
main/resources/static/basic/hello-form.html
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Title</title>
</head><body>
<form action="/request-param-v1" method="post">
username: <input type="text" name="username" />
age: <input type="text" name="age" />
<button type="submit">전송</button>
</form>
</body>
</html>
POST Form 실행
http://localhost:8080/basic/hello-form.html
username=kim, age=123
참고
Jar
를 사용하면webapp
경로를 사용할 수 없다. 이제부터 정적 리소스도 클래스 경로에 함께 포함해야 한다.
스프링이 제공하는 @RequestParam
을 사용하면 요청 파라미터를 매우 편리하게 사용할 수 있다.
/**
* @RequestParam 사용
* - 파라미터 이름으로 바인딩
* @ResponseBody 추가
* - view 조회를 무시하고, HTTP message body에 직접 해당 내용 입력
*/
@ResponseBody
@RequestMapping("/request-param-v2")
public String requestParamV2(
@RequestParam("username") String memberName,
@RequestParam("age") int memberAge) {
log.info("username={}, age={}", memberName, memberAge);
return "ok";
}
@RequestParam
: 파라미터 이름으로 바인딩
@ResponseBody
: View 조회를 무시하고, HTTP messgae body에 직접 해당 내용 입력
@RequestParam의 name(value)
속성이 파라미터 이름으로 사용
💬 @Controller + String return "ok"; => ok라는 이름의 뷰를 찾는다.
✔️ 해결법1. 클래스 단위 애노테이션 @Controller -> @RestController로 수정
✔️ 해결법2. 메서드 위에 @ResponseBody 추가.
=> String문자를 http 응답메세지에 콱 넣어서 반환함. (@RestController와 같은 효과)
/**
* @RequestParam 사용
* HTTP 파라미터 이름이 변수 이름과 같으면 @RequestParam(name="xx") 생략 가능
*/
@ResponseBody
@RequestMapping("/request-param-v3")
public String requestParamV3(
@RequestParam String username,
@RequestParam int age) {
log.info("username={}, age={}", username, age);
return "ok";
}
변수명을 통일하고, @RequestParam
애노테이션 뒤의 (name=username)
부분을 생략할 수 있다.
/**
* @RequestParam 사용
* String, int 등의 단순 타입이면 @RequestParam도 생략 가능
*/
@ResponseBody
@RequestMapping("/request-param-v4")
public String requestParamV4(String username, int age) {
log.info("username={}, age={}", username, age);
return "ok";
}
String, int, Integer
등의 단순 타입이면 @RequestParam
도 생략 가능.
주의
@RequestParam
애노테이션을 생략하면 스프링MVC는 내부에서required=false
를 적용한다.
참고
이렇게 애노테이션을 완전히 생략해도 되는데, 너무 없는것도 약간 과할 수 있다.@RequestParam
이 있으면 명확하게 요청 파라미터에서 데이터를 읽는 다는 것을 알 수 있다.
/**
* @RequestParam.required
* /request-param-required -> username이 없으므로 예외
*
* 주의!
* /request-param-required?username= -> 빈문자로 통과
*
* 주의!
* /request-param-required
* int age -> null을 int에 입력하는 것은 불가능, 따라서 Integer 변경해야 함
* (또는 다음에 나오는 defaultValue 사용)
*/
@ResponseBody
@RequestMapping("/request-param-required")
public String requestParamRequired(
@RequestParam(required = true) String username,
@RequestParam(required = false) int age) {
log.info("username={}, age={}", username, age);
return "ok";
}
@RequestParam.required
요청1.
/request-param?age=20
요청
username
이 없으므로 400예외가 발생한다.400 : 전송된 Request 문제 (Bad Request exception)
양방향 스펙을 설정할때 특정 변수를 필수라고 정한 값인데, 해당 값을 전송하지 않으면 BadRequest를 보내는것이 맞다.(HTTP 스펙을 준수하지 않은 것이므로)
요청2.
/request-param?username=kim
요청
defaultValue
사용.500 : 서버단의 문제 ( 권한 문제 )
요청3.
/request-parma?username=
요청
/**
* @ReuqestParam
* - defaultValue 사용
*
* 참고: defaultValue는 빈 문자의 경우에도 적용
* /request-param-default?username=
*/
@ResponseBody
@RequestMapping("/request-param-default")
public String requestParamDefault(
@RequestParam(required = true, defaultValue = "guest") String username,
@RequestParam(required = false, defaultValue = "-1") int age) {
log.info("username={}, age={}", username, age);
return "ok";
}
defaultValue
로 설정한 값을 사용하므로 값을 사용하므로, Required
의미가 없다. /**
* @RequestParam Map, MultiValueMap
* Map(key=value)
* MultiValueMap(key=[value1, value2, ...]) ex) (key=userIds, value=[id1, id2])
*/
@ResponseBody
@RequestMapping("/request-param-map")
public String requestParamMap(@RequestParam Map<String, Object> paramMap) {
log.info("username={}, age={}", paramMap.get("username"), paramMap.get("age"));
return "ok";
}
@ResponseBody
@RequestMapping("/request-param-multimap")
public String requestParamMap(@RequestParam MultiValueMap<String, Object> paramMap) {
log.info("username={}, age={}", paramMap.get("username"), paramMap.get("age"));
return "ok";
}
@RequestParam Map
Map(key=value)
@RequestParam MultiValueMap
MultiValueMap(key=[value1, value2, ...] ex) (key=userIds, value=[id1, id2])
파라미터의 값이 1개가 확실하다면 Map
을 사용해도 되지만, 그렇지 않다면 MultiValueMap
을 사용하자.
파라미터 값은 보통 하나만 넘긴다. 여러개 넘기는경우 거의 없다. 없는게 좋을거 같다 내가 생각해도.
실행
http://localhost:8080/request-param-map?username=1&username=2
http://localhost:8080/request-param-multimap?username=1&username=2
실제 개발을 하면 요청 파라미터를 받아서 필욯나 객체를 만들고 그 객체에 값을 넣어주어야 한다. 보통 다음과 같이 코드를 작성할 것이다.
@RequestParam String username;
@RequestParma int age;
HelloData data = new HelloData();
data.serUesrname(username);
data.setAge(age);
스프링은 이 과정을 완전히 자동화해주는 @ModelAttribute
기능을 제공한다.
먼저 요청 파라미터를 바인딩 받을 객체를 만들자.
HelloData
package hello.springmvc.basic;
import lombok.Data;
@Data
public class HelloData {
private String username;
private int age;
}
롬복 @Data
@Getter
, @Setter
, @ToString
, @EqualsAndHashCode
, @RequiredArgsConstructor
를 자동으로 적용해준다. /**
* @ModelAttribute 사용
* 참고: model.addAttribute(helloData) 코드도 함께 적용됨, 뒤에 model을 설명할 때 자세히 설명
*/
@ResponseBody
@RequestMapping("/model-attribute-v1")
public String modelAttributeV1(@ModelAttribute HelloData helloData) {
log.info("username={}, age={}", helloData.getUsername(), helloData.getAge());
return "ok";
}
마치 마법처럼 HelloData
객체가 생성되고, 요청 파라미터의 값도 모두 들어가있다.
스프링 MVC는 @ModelAttribute
가 있으면 다음을 실행한다.
HelloData
객체를 생성한다.HelloData
객체의 프로퍼티를 찾는다. 그리고 해당 프로퍼티의 setter를 호출해서 파라미터의 값을 입력(바인딩)한다.username
이면 setUsername()
메서드를 찾아서 호출하면서 값을 입력한다. 프로퍼티
객체에 getUsername()
, serUsername()
메서드가 있으면, 이 객체는 username
이라는 프로퍼티를 가지고 있다. username
프로퍼티의 값을 변경하면 setUsername()
이 호출되고, 조회하면 getUsername()
이 호출된다.
바인딩 오류
age=age
처럼 숫자가 들어가야 할 곳에 문자를 넣으면 BindException
이 발생한다. 이런 바인딩 오류를 처리하는 방법은 검증 부분에서 다룬다.
정식 코드는 20, 30%정도. 오류 검증및 처리가 훨씬 오래걸린다. 중요하게 다뤄야 하는 부분이라 나중에 따로 자세히 설명할 것!
?age=abc
호출바인딩(binding)
컴퓨터 프로그래밍에서 각종 값들이 확정되어 더 이상 변경할 수 없는 구속(bind)상태가 되는 것. 프로그램 내에서 변수, 배열, 라벨, 절차 등의 명칭, 즉 식별자(identifier)가 그 대상인 메모리 주소, 데이터형 또는 실제 값으로 배정되는 것이 이에 해당되며, 원시 프로그램의 컴파일링 또는 링크 시에 확정되는 바인딩을 정적 바인딩(static binding)이라 하고, 프로그램 실행 과정에서 바인딩 되는 것을 동적 바인딩(dynamic binding)이라고 한다. 프로그래머가 코딩을 해서 컴파일을 하게 되면 프로그래머가 값을 변경할 수 없는 상태가 되는데, 이걸 바인딩이라고 한다. 메모리에 값을 할당하는 것 또한 바인딩이라고 할수 있다. 쉽게 말하자면 바인딩이란 프로그램의 어떤 기본 단위가 가질 수 있는 구성요소의 구체적인 값, 성격을 확정하는 것을 말한다.
/**
* @ModelAttribute 생략 가능
* String, int같은 단순 타입 = @RequestParam
* 'argumet resolver'로 지정해둔 타입 외 = @ModelAttribute
*/
@ResponseBody
@RequestMapping("/model-attribute-v2")
public String modelAttributeV2(HelloData helloData) {
log.info("username={}, age={}", helloData.getUsername(), helloData.getAge());
return "ok";
}
@ModelAttribute
는 생략할 수 있다. 그런데 @RequestParam
도 생략할 수 있으니 혼란이 발생할 수 있다.
스프링은 해당 생략시 다음과 같은 규칙을 적용한다.
String, int, Integer
같은 단순타입 = @RequestParam
@ModelAttribute
(argument resolver로 지정해둔 타입 외)HttpServletResponse
도 argument resolver 설정 예시 중 하나. argument resolver를 통해 특정 객체들이 예약이 되어 있다. 내가 직접 만든 객체들은 다 자동으로 modelAttribute로 적용이 됨. 당연히 내가 argument resolver에 추가하면 적용되지 않는다.
ModelAttribute(name="")
으로 name도 적용가능. 그러나 요청이 올때 name과 관련없이, 객체의 멤버변수명과 일치하기만 하면 된다. name 적용 대상은 주로 view에 모델 넘길때 사용.
지금까지 요청 파라미터를 알아본것. 이제 http body에 직접 데이터가 넘어온 것을 알아볼 예정.
서블릿에서 학습한 내용을 떠올려보자.
HTTP message body에 데이터를 직접 담아서 요청
요청 파라미터와 다르게, HTTP 메시지 바디를 통해 데이터가 직접 데이터가 넘어오는 경우는 @RequestParam
, @ModelAttribute
를 사용할 수 없다.(물론 HTML Form 형식으로 전달되는 경우는 요청 파라미터로 인정된다.)
먼저 가장 단순한 텍스트 메시지를 HTTP 메시지 바디에 담아서 전송하고, 읽어보자.
HTTP 메시지 바디의 데이터를 InputStream
을 사용해서 직접 읽을 수 있다.
@Slf4j
@Controller
public class RequestBodyStringController {
@PostMapping("/request-body-string-v1")
public void requestBodyString(HttpServletRequest request, HttpServletResponse response) throws IOException {
ServletInputStream inputStream = request.getInputStream();
String messageBody = StreamUtils.copyToString(inputStream, StandardCharsets.UTF_8);
log.info("messageBody={}", messageBody);
response.getWriter().write("ok");
}
}
Postman을 통해서 테스트해보자.
POST 설정 + http://localhost:8080/request-body-string-v1
Body -> row, Text 선택
앞서 서블릿을 통해 우리가 직접 바디를 긁어오던 코드와 똑같다. 이제이걸 스프링이 어떻게 사용하기 용이하게 변경시켜주는지 살펴보자
/**
* InputStream(Reader): HTTP 요청 메시지 바디의 내용을 직접 조회
* OutputStream(Writer): HTTP 응답 메시지의 바디에 직접 결과 출력
*/
@PostMapping("/request-body-string-v2")
public void requestBodyStringV2(InputStream inputStream, Writer responseWriter) throws IOException {
String messageBody = StreamUtils.copyToString(inputStream, StandardCharsets.UTF_8);
log.info("messageBody={}", messageBody);
responseWriter.write("ok");
}
스프링 MVC는 다음 파라미터를 지원한다.
InputStream(Reader)
: HTTP 요청 메시지 바디의 내용을 직접 조회OutputStream(Writer)
: HTTP 응답 메시지의 바디에 직접 결과 출력 /**
* HttpEntity: Http header, body 정보를 편리하게 조회
* - 메시지 바디 정보를 직접 조회(@RequetParma X, @ModelAttribute X)
* - HttpMessageConverter 사용 -> StringHttpMessageConverter 적용
*
* 응답에서도 HttpEntity 사용 가능
* - 메시지 바디 정보 직접 반환(view 조회 X)
* - HttpMessageConverter 사용 -> StringHttpMessageConverter 적용
*/
@PostMapping("/request-body-string-v3")
public HttpEntity<String> requestBodyStringV3(HttpEntity<String> httpEntity) throws IOException {
String messageBody = httpEntity.getBody(); //Http의 변환된 바디를 꺼낼 수 있음.
log.info("messageBody={}", messageBody);
return new HttpEntity<>("ok");
// 파라미터로 bodymessage를 넣을 수 있음
}
스프링 MVC는 다음 파라미터를 지원한다.
HttpEntity : HTTP header, body 정보를 편리하게 조회
HttpEntity 는 응답에도 사용 가능
HttpEntity
를 상속받은 다음 객체들도 같은 기능을 제공한다.
RequestEntity: HttpMethod, url정보가 추가, 요청에서 사용
ResponseEntity: HTTP 상태코드 설정 가능, 응답에서 사용
@PostMapping("/request-body-string-v3")
public HttpEntity<String> requestBodyStringV3(RequestEntity<String> requestEntity) throws IOException {
String messageBody = requestEntity.getBody(); //Http의 변환된 바디를 꺼낼 수 있음.
log.info("messageBody={}", messageBody);
return new ResponseEntity<>("ok", HttpStatus.CREATED); // 상태코드를 넣을 수 있음.
}
참고
스프링 mvc 내부에서 http 메시지 바디를 읽어서 문자나 객체로 변환해서 전달해주는데, 이때 HTTP 메시지 컨버터(HttpMessageConverter)라는 기능을 사용한다. 이것은 조금 뒤에 HTTP 메시지 컨버터에서 자세히 설명한다.
종결버전. 이렇게 많이 쓴다.
/**
* @RequestBody
* - 메시지 바디 정보를 직접 조회(@RequestParam X, @ModelAttribute X)
* - HttpMessageConverter 사용 ->StringHttpMessageConverter 적용
*
* @ResponseBody
* - 메시지 바디 정보 직접 반환(view 조회X)
* - HttpMessageConverter 사용 -> StringHttpMessageConverter 적용
*/
@ResponseBody
@PostMapping("/request-body-string-v4")
public String requestBodyStringV4(@RequestBody String messageBody) throws IOException {
log.info("messageBody={}", messageBody);
return "ok";
}
@RequestBody
@RequestBody
를 사용하면 HTTP 메시지 바디 정보를 편리하게 조회할 수 있다. 참거로 헤더 정보가필요하다면 HttpEntity
를 사용하거나 @ReqjestHeader
를 사용하면 된다. 이렇게 메시지 바디를 직접 조회하는 기능은 요청파라미터를 조회하는 @RequestParam
, @ModelAttribute
와는 전혀 관계가 없다.
요청 파라미터 vs Http 메시지 바디
@RequestParam
, @ModelAttribute
@RequestBody
요청 파라미터: 1.(get의 queryParameter) or 2.(POST로 HTML Form방식.
content-type:x-www-urlencoded
)으로 전송.
그 외는 전부 HttpEntity 같은것들을 이용해서 body에서 직접 꺼내야 함.
@ResponseBody
@ResponseBody
를 사용하면 응답결과를 HTTP 메시지 바디에 직접 담아서 전달할 수 있다. 물론 이 경우에도 view를 사용하지 않는다.
이번에는 HTTP API에서 주로 사용하는 JSON 데이터 형식을 조회해보자.
기존 서블릿에서 사용했던 방식과 비슷하게 시작해보자.
package hello.springmvc.basic.request;
import com.fasterxml.jackson.databind.ObjectMapper;
import hello.springmvc.basic.HelloData;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Controller;
import org.springframework.util.StreamUtils;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.ResponseBody;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
/**
* {"username":"hello", "age":20}
* content-type:application/json
*/
@Slf4j
@Controller
public class RequestBodyJsonController {
private ObjectMapper objectMapper =new ObjectMapper();
@PostMapping("/request-body-json-v1")
public void requestBodyJsonV1(HttpServletRequest request, HttpServletResponse response) throws IOException {
ServletInputStream inputStream = request.getInputStream();
String messageBody = StreamUtils.copyToString(inputStream, StandardCharsets.UTF_8);
log.info("messageBody={}", messageBody);
HelloData helloData = objectMapper.readValue(messageBody, HelloData.class);
log.info("username={}, age={}", helloData.getUsername(), helloData.getAge());
response.getWriter().write("ok");
}
}
기존에 서블릿에서 학습한것과 다를게 없다.
ojbectMapper
를 사용해서 자바 객체ㅗ 변환한다.Postman으로 테스트
{"username":"hello", "age":20}
/**
* @RequestBody
* HttpMessageConverter 사용 -> StringHttpMessageConverter 적용
*
* @ResponseBody
* - 모든 메서드에 @ResponseBody 적용
* - 메시지 바디 정보 직접 반환(view 조회X)
* - HttpMessageConverter 사용 -> StringHttpMessageConverter 적용
*/
@ResponseBody
@PostMapping("/request-body-json-v2")
public String requestBodyJsonV2(@RequestBody String messageBody) throws IOException {
log.info("messageBody={}", messageBody);
HelloData helloData = objectMapper.readValue(messageBody, HelloData.class);
log.info("username={}, age={}", helloData.getUsername(), helloData.getAge());
return "ok";
}
이전 장처럼 기존의 코드에서 문자바꾸기 적용
@RequestBody
를 사용해서 HTTP 메시지에서 데이터를 꺼내고 messageBody에 저장한다.messageBody
를 ObjectMapper
를 통해서 자바 객체로 변환한다. 문자로 변환하고 다시 json으로 변환하는 과정이 불편하다. @ModelAttribute처럼 한번에 객체로 변환할 수는 없을까?
아, 이거 내가 굳이 objectMapper로 바꿔줘야 해? @ModelAttribute 할 때 되겨 편했잖아. 이건 그런거 없나? 됩니다요!v3
@RequestBody 객체 파라미터
/**
* @RequestBody 생략 불가능(@ModelAttribute가 적용되어 버림)
* HttpMessageConverter 사용 -> MappingJacson2HttpMessageConverter
* (content-type:application/json)
*/
@ResponseBody
@PostMapping("/request-body-json-v3")
public String requestBodyJsonV3(@RequestBody HelloData helloData) throws IOException {
log.info("username={}, age={}", helloData.getUsername(), helloData.getAge());
return "ok";
}
@RequestBody HelloData data
@RequestBody
에 직접 만든 객체를 지정할 수 있다.HttpEntity
, @ResponseBody
를 사용하면 HTTP메시지 컨버터가 HTTP 메시지 바디의 내용을 우리가 원하는 문자나 객체 등으로 변환해준다. HTTP 메시지 컨버터는 문자 뿐만 아니라 JSON도 객체로 변환해주는데, 우리가 방금 V2에서 했던 작업을 대신 처리해준다. 자세한 내용은 뒤에 HTTP 메시지 컨버터에서 다룬다.
HttpEntity<HelloData>
이렇게 쓰면 HelloData 객체에 맞춰 반환해준다. HttpMessageConverter가 JSON이면 HelloData helloData = objectMapper.readValue(messageBody, HelloData.class);
코드를 대신 실행해주고 그렇게 생성된 객체를 helloData에 대신 넣어주는 것. 문자를 컨버터하는 컨버터도 있고 json을 컨버터해주는것이 또 따로있다.
@RequestBody는 생략 불가능
@MocelAttribute
에서 학습한 내용을 떠올려보자.
스프링은 @ModelAttribute
, @RequestParam
과 같은 해당 애노테이션을 생략시 다음과 같은 규칙을 적용한다.
String, int, Integer
과 같은 단순타입 = @RequestParam
@ModelAttribute
(argument resolver로 지정해둔 타입 외)@ModelAttribute
가 붙어버린다. 따라서 생략하면 HTTP 메시지 바디가 아니라 요청 파라미터를 처리하게 된다.주의
HTTP 요청시에 content-type이 applicatoin/json인지 꼭! 확인해야 한다. 그래야 JSON을 처리할 수 있는 HTTP 메시지 컨버터가 실행된다.
물론 앞서 배운 것과 같이 HttpEntity를 사용해도 된다.
@ResponseBody
@PostMapping("/request-body-json-v4")
public String requestBodyJsonV4(HttpEntity<HelloData> helloData) throws IOException {
HelloData data = helloData.getBody();
// getBody하면 제네릭 타입이 반환됨.
log.info("username={}, age={}", data.getUsername(), data.getAge());
return "ok";
}
/**
* @RequestBody 생략 불가능(@ModelAttirbute가 적용되어 버림)
* HttpMessageConverter 사용 -> MappingJacson2HttpMessageConverter
* (content-type:applicaion/json)
*
* @ResponseBody 적용
* - 메시지 바디 정보 직접 반환(view 조회X)
* - HttpMessageConverter 사용 -> MappingJackson2HttpMessageConverter적용
* (Accept: application/json)
*/
@ResponseBody
@PostMapping("/request-body-json-v5")
public HelloData requestBodyJsonV5(@RequestBody HelloData helloData) throws IOException {
log.info("username={}, age={}", helloData.getUsername(), helloData.getAge());
return helloData;
}
@ResponsBody
응답의 경우에도 @ResponseBody
를 사용하면 해당 객체를 HTTP 메시지 바디에 직접 넣어줄 수 있다. 물론 이 경우에도 HttpEntity
를 사용해도 된다.
@RequestBody
요청
@RequestBody
응답
응답 데이터는 이미 앞에서 일부 다룬 내용들이지만, 응답 부분에 초점을 맞추어서 정리해보자.
스프링(서버)에서 응답 데이터를 만드는 방법은 크게 3가지이다.
정적 리소스
뷰 템플릿 사용
HTTP 메시지 사용
스프링부트는 클래스패스의 다음 디렉토리에 있는 정적 리소스를 제공한다.
/static, /public, /resources, /META-INF/resources/
src/main/resources
는 리소스를 보관하는 곳이고, 또 클래스패스의 시작 경로이다. 따라서 다음 디렉토리에 리소스를 넣어두면 스프링부트가 정적 리소스로 서비스를 제공한다.
static 안에 들어있는 파일들은 스프링 부트가 정적 리소스로 제공한다.
public등 위에 나열한 이름들은 다 가능. 해당 주소 안에 있으면 웹 브라우저를 통해 클라이언트가 실행 가능하다.
정적 리소스 경로
src/main/reources/static
다음 경로에 파일이 들어있으면 (src/main/resources/static/basic/hello-form.html
) 웹 브라우저에서 다음과 같이 실행하면 된다. http://localhost:8080/basic/hello-form.html
호출.
정적 리소스는 해당 파일을 변경 없이 그대로 서비스하는 것이다.
뷰 템플릿을 거쳐서 HTML이 생성되고, 뷰가 응답을 만들어서 전달한다.
일반적으로 HTML을 동적으로 생성하는 용도로 사용하지만, 다른 것들도 가능하다. 뷰 템플릿이 만들 수 있는 것이라면 뭐든지 가능하다.
스프링 부트는 기본 뷰 템플릿 경로를 제공한다.
뷰 템플릿 경로
src/main/resources/templates
뷰 템플릿 생성
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<p th:text="${data}">empty</p>
</body>
</html>
<html xmlns:th="http://www.thymeleaf.org">
: 기본으로 넣는다.
<p th:text="${data}">empty</p>
:<th:text>
코드 사용시 model에서 data라는 키의 값을 꺼내서 empty 부분을 data의 값으로 치환해서 보여준다.
ResponseViewController - 뷰 템플릿을 호출하는 컨트롤러
package hello.springmvc.basic.response;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.servlet.ModelAndView;
@Controller
public class ResponseViewController {
@RequestMapping("/response-view-v1")
public ModelAndView responseViewV1() {
ModelAndView mav = new ModelAndView("response/hello")
.addObject("data", "hello");
return mav;
}
// @ResponseBody
@RequestMapping("/response-view-v2")
public String responseViewV2(Model model) {
model.addAttribute("data", "hello!");
return "response/hello";
}
@RequestMapping("/response/hello")
public void responseViewV3(Model model) {
model.addAttribute("data", "hello!");
}
}
v1
ModelAndView
에 논리주소와 객체를 넣고 반환.
v2: String을 반환하는 경우 - View 혹은 HTTP 메시지
@Controller
+ String 반환 = String 값이 뷰의 논리적이름이 된다.
따라서 response/hello
로 뷰 리졸버가 실행되어서 뷰를 찾고, 렌더링 한다.
만약 @ResponseBody
가 붙으면 뷰 리졸버를 실행하지 않고, HTTP 메시지 바디에 직접 response/hello
라는 문자가 응답된다.
@Controller
를 사용하고, HttpSErvletResponse
,OutputStream(Writer)
같은 HTTP 메시지 바딜르 처리하는 파라미터가 없으면 요청 URL을 참고해서 논리 뷰 이름으로 사용/reponse/hello
templates/response/hello.html
@ResponseBody
, HttpEnttity
를 사용하면, 뷰 템플릿을 사용하는 것이 아니라, HTTP 메시지 바디에 직접 응답 데이터를 출력할 수 있다.다음 라이버리를 추가하면 스프링 부트가 자동으로 ThymeleafViewReslover
와 필요한 스프링 빈을 등록한다. 그리고 다음 설정도 사용한다. 이 설정은 기본 값이기 때문에 변경이 필요할 때만 설정하면 된다.
build.gradle
`implementation 'org.springframework.boot:spring-boot-starter-thymeleaf'`
application.properties
spring.thymeleaf.prefix=classpath:/templates/
spring.thymeleaf.suffix=.html
타임리프 설정 기본값은 입력 안해도 설정됨. 변경하고 싶은 경우에만 설정값을 직접 입력하고 수정해주면 된다.
참고
스프링부트의 타임리프 관련 추가 일정은 다음 공식 사이트를 참고하자.(페이지 안에서 thymeleaf 검색)
https://docs.spring.io/spring-boot/docs/2.4.3/reference/html/appendix-application-properties.html#common-application-properties-templating
HTTP API를 제공하는 경우에는 HTML이 아니라 데이터를 전달해야 하므로, HTTP 메시지 바디에 JSON 같은 형식으로 데이터를 실어 보낸다. HTTP 요청에서 응답까지 대부분 다루었으므로 이번에는 정리를 해보자.
참고
HTML이나 뷰 템플릿을 사용해도 HTTP 응답 메시지 바디에 HTML 데이터가 담겨서 전달된다. 여기서 설명하는 내용은 정적 리소스나 뷰 템플릿을 거치지 않고, 직접 HTTP 응답 메시지를 전달하는 경우를 말한다.
package hello.springmvc.basic.response;
import hello.springmvc.basic.HelloData;
import lombok.extern.slf4j.Slf4j;
import org.apache.coyote.Response;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.ResponseStatus;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
@Slf4j
@Controller
//@ResponseBody
@RestController
public class ResponseBodyController {
// 기본 방식
@GetMapping("/response-body-string-v1")
public void responseBodyV1(HttpServletResponse response) throws IOException {
response.getWriter().write("ok");
}
// 좀더 단순하게 스프링에서 제공하는 방식
/**
* HttpEntity, ResponseEntity(Http Status 추가)
*/
@GetMapping("/response-body-string-v2")
public ResponseEntity responseBodyV2() {
return new ResponseEntity<>("ok", HttpStatus.OK);
}
// 거기서 더 편하게
@GetMapping("/response-body-string-v3")
@ResponseBody
public String responseBodyV3() {
return "ok";
}
// json은 어떻게 처리할까?
@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);
}
@ResponseStatus(HttpStatus.OK)
@ResponseBody
@GetMapping("/response-body-json-v2")
public HelloData responseBodyJsonV2() {
HelloData helloData = new HelloData();
helloData.setUsername("userA");
helloData.setAge(20);
return helloData;
}
// 이 경우에는 상태메시지를 내가 바꿀 수 없음. => @ResponseStatus 기능을 제공.
}
resonseBodyV1
서블릿을 직접 다룰 때 처럼 HttpServletResponse 객체를 통해서 HTTP 메시지바디에 직접 ok
응답 메시지를 전달한다.
response.getWriter().writer("ok");
responseBodyV2
ResponseEntity
엔티티는 HttpEntity
를 상속받았는데, HttpEntity는 HTTP 메시지의 헤더, 바디 정보를 가지고 있다. ResponseEntity
는 여기에 더해서 HTTP 응답 코드를 설정할 수 있다.
HttpStatus.CREATED
로 변경하면 201 응답이 나가는것을 확인할 수 있다.
responseBodyV3
@ResponseBody
를 사용하면 view를 사용하지 않고, HTTP 메시지 컨버터를 통해서 HTTP 메시지를 직접 입력할 수 있다. ResponseEntity
도 동일한 방식으로 동작한다.
< JSON의 경우 >
responseBodyJsonV1
ResponseEntity
를 반환한다. HTTP 메시지 컨버터를 통해서 JSON 형식으로 변환되어 반환된다.
responseBodyJsonV2
ResponseEntity
는 HTTP 응답 코드를 설정할 수 있는데, @ResponseBody
를 사용하면 이런 것을 설정하기 까다롭다. @ResponseStatus(HttpStatus.OK)
애노테이션을 사용하면 응답 코드도 설정할 수 있다.
물론 애노테이션이기 때문에 코드를 동적으로 변경할 수는 없다. 프로그램 조건에 따라서 동적으로 변경하려면 ResonseEntity
를 사용하면 된다.
@ResponseBody
: 클래스 단위로 붙일 수 있음. 그럼 모든 메서드에 다 적용된다.@ResController
: @Controller
+ @ResponseBody
@Controller
대신에 @RestController
애노테이션을 사용하면, 해당 컨트롤러에 모두 @ResponseBody
가 적용되는 효과가 있다. 따라서 뷰 템플릿을 사용하는 것이 아니라, HTTP 메시지 바디에 직접 데이터를 입력한다. 이름 그대로 Rest API(HTTP API)를 만들때 사용하는 컨트롤러이다.뷰 템플릿으로 HTML을 생성해서 응답하는 것이 아니라, HTTP API처럼 JSON데이터를 HTTP 메시지 바디에서 직접 읽거나 쓰는 경우 HTTP 메시지 컨버터를 사용하면 편리하다.
과거 서블릿에서 response.getWriter.write("ok")
이런식으로 구현하는거 너무 힘들고 번거로웠음. 그런 부분들을 HTTP 메시지 컨버터가 대신 수행해준다!
HTTP 메시지 컨버터를 설명하기 전에 잠깐 과거로 돌아가서 스프링 입문 강의에서 설명했던 내용을 살펴보자.
웹브러우저에서 요청이 오면 컨트롤러가 호출되고, @ResponseBody가 있으면 HttpMessageConverter가 작동하여 json으로 나갈지, String으로 나갈지 결정해서 hello return값을 반환함(여기선 객체로 반환)
@ResponseBody
를 사용
viewResolver
대신에 HttpMessageConverter
가 동작StringHttpMessageConverter
MappingJackson2HttpMessageConverter
참고: 응답의 경우 클라이언트의 HTTP Accept 헤더와 서버의 컨트롤러 반환 타입 정보 두을 조합해서 (사실은 몇개 더 고려함)
HttpMessageConverter
가 선택된다.
이제 다시 돌아와서,
스프링 MVC는 다음의 경우에 HTTP 메시지 컨버터를 적용한다.
@RequestBody
, HttpEntity(RequestEntity)
@ResponseBody
, HttpEntity(ResponseEntity)
HTTP 메시지 컨버터는 HTTP 요청, HTTP 응답 둘 다 사용된다. (양방향)
http 메시지 컨버터는 인터페이스로 되어 있음. (json처리, String처리 등등 여러가지 종류가 있음. 이 여러종류들을 아우르는 부모 인터페이스가 존재)
org.springframework.http.converter.HttpMessageConverter
package org.springframework.http.converter;
public interface HttpMessageConverter<T> {
boolean canRead(Class<?> clazz, @Nullable MediaType mediaType);
boolean canWrite(Class<?> clazz, @Nullable MediaType mediaType);
List<MediaType> getSupportedMediaTypes();
T read(Class<? extends T> clazz, HttpInputMessage inputMessage)
throws IOException, HttpMessageNotReadableException;
void write(T t, @Nullable MediaType contentType, HttpOutputMessage
outputMessage)
throws IOException, HttpMessageNotWritableException;
}
HTTP 메시지 컨버터는 HTTP 요청, HTTP 응답 둘 다 사용된다.
canRead()
, canWrite()
: 메시지 컨버터가 해당 클래스, 미디어 타입을 지원하는지 체크read()
, write()
: 메시지 컨버터를 통해서 메시지를 읽고 쓰는 기능.스프링 부트는 기본적으로 올라올때 몇가지 메시지 컨버터들을 등록해놓음.
(일부 생략)
0 = ByteArrayHttpMessageConverter
1 = StringHttpMessageConverter
2 = MappingJackson2HttpMessageConverter
스프링 부트는 다양한 메시지 컨버터를 제공하는데, 대상 클래스 타입과 미디어 타입 둘을 체크해서 사용 여부를 결정한다. 만약 만족하지 않으면 다음 메시지 컨버터로 우선순위가 넘어간다.
미디어 타입 : Http 요청인 경우에는 Content-Type.
몇가지 주요한 메시지 컨버터를 알아보자.
ByteArrayHttpMessageConverter
byte[]
, 미디어 타입 */*
(다 된다는 의미)@RequestBody byte[] data
@ResponseBody return byte[]
쓰기 미디어타입 applicaion/octet-stream
StringHttpMessageConverter
String
, 미디어 타입 */*
@RequestBody String data
@ResponseBody return "ok
쓰기 미디어타입 text/plain
MappingJackson2HttpMessageConverter
: application/json
HashMap
, 미디어 타입 application/json
관련@RequestBody HelloData data
@ResponseBody return helloData
쓰기 미디어타입 application/json
관련항상 http 메시지 바디에 데이터가 있으면 컨텐트 타입을 지정해줘야 한다. 요청메시지든 응답 메시지든.
StringHttpMessageConverter
content-type: applicaion/json
@RquestMapping
void hello(@RequestBody String data) {}
이런 요청이 오면,
@Request String
=> 0: byte 아님. 1: String 맞음!StringHttpMessageConverter
의 미디어 타입이 다 수용가능한 */*
이므로 커버 가능함.💬 어? Content-Type이 application/json인데요?
✔️@RequestBody String
인 시점에서MappingJackson2HttpMessageConverter
은 우선순위가 밀립니다. 우선순위 높은StringHttpMessageConverter
가 먼저 채택되게 됨니다.
MappingJackson2HttpMessageConverter
content-type: application/json
@RequestMapping
void hello(@RequestBody Hellodata data) {}
MappingJackson2HttpMessageConverter
의 미디어 타입: application/json
으로 일치함.MappingJackson2HttpMessageConverter
채택?
content-type: text/html
@RequestMapping
void hello(@RequestBody HelloData data) {}
MappingJackson2HttpMessageConverter
의 미디어 타입 applicaion/json
이 아님. @RequestBody
, HttpEntity
파라미터를 사용한다. canRead()
를 호출한다.@RequestBody
의 대상 클래스( byte[]
, String
, HelloData
) text/plain
, application/json
, */*
canRead()
조건을 만족하면 read()
를 호출해서 객체 생성하고, 반환한다.@ResponseBody
, HttpEntity
로 값이 반환된다.canWrite()
를 호출한다.@RequestMapping
의 produces
)text/plain
, application/json
, */*
canWrtie()
조건을 만족하면 write()
를 호출해서 HTTP 응답 메시지 바디에 데이터를 생성한다.RequestMappingHandlerAdapter. 컨트롤러가 처리해주는 바로 그것.
그렇다면 HTTP 메시지 컨버터는 스프링 MVC 어디쯤에서 사용되는 것일까? 다음 그림에서는 보이지 않는다.
springMVC 구조
모든 비밀은 애노테이션 기반의 컨트롤러, 그러니까 @RequestMaping
을 처리하는 핸들러 어댑터인 RequestMappingHandlerAdapter
(요청 매핑 핸들러 어댑터)에 있다.
핸들러 어댑터-> 핸들러. 이 부분에 비밀이 있다!!!
지금까지 당연하게 생각했던 것들에 한 번 의문을 가져보자. 앞서 진행했던 코드다. requestBodyJsonV1
메서드의 파라미터인 HttpServletRequest
는 누가 넣어줬을까? @ResponseBody String messageBody
는? 누가 당연하다는 듯 byte를 String으로 바꿔서 넣어줬을까? 나는 넣은적 없다. 당연한 것은 없음에도 불구하고 나는 당연하다는 듯이 파라미터들을 요구하고, 파라미터의 변형을 요구했다. 그리고 누군가 소리소문 없이 그 일들을 수행하고 있었다! 누가 일일이 파라미터들을 정제하고 삽입해주었을까?
=> 바로 Argument Resolver 다!!
ReuqestMappingHandlerAdatper
는 ArgumentResolver
를 호출해서 HttpServletRequest
객체 가져다 줄수 있어? HttpServletResponse
도 가져다 줄수있어? 하고 물어봄. 그럼 ArgumentResolver
가 객체를 가져다 줌. 특히 @RequestBody String messageBody
같은거 ArgumentResolver
가 다 만들어서 가져다줌. 그래서 다 가져다 주면 핸들러어댑터가 "나 준비 다 됐어!"하며 핸들러를 호출하고 파라미터를 건내줌.
생각해보면, 애노테이션 기반의 컨트롤러는 매우 다양한 파라미터를 사용할 수 있었다.
HttpServletRequest
,Model
은 물론이고, @RequestParam
, @ModelAttribute
같은 애노테이션 그리고 @RequestBody
, HttpEntity
같은 http 메시지를 처리하는 부분까지 매우 큰 유연함을 보여주었다. 이렇게 파라미터를 유연하게 처리할 수 있는 이유가 바로 ArgumentResolver
덕분이다.
애노테이션 기반 컨트롤러를 처리하는 ReuqestMappingHandlerAdapter
는 바로 이 ArgumentResolver
를 호출해서 컨트롤러(핸들러)가필요로 하는 다양한 파라미터의 값(객체)을 생성한다. 그리고 이렇게 파라미터의 값이 모두 준비되면 컨트롤러를 호출하면서 값을 넘겨준다.
스프링은 30개가 넘는 ArgumentResolver
를 기본으로 제공한다. 어떤 종류들이 있는지 살짝 코드로 확인만 해보자.
참고: 가능한 파라미터 목록은 다음 공식 메뉴얼에서 확인할 수 있다.
https://docs.spring.io/spring-framework/docs/current/reference/html/web.html#mvc-ann-arguments
정확히는 HandlerMethodArgumentResolver
인데 줄여서 ArgumentResolver
라고 부른다.
public interface HandlerMethodArgumentResolver {
boolean supportsParameter(MethodParameter parameter);
@Nullable
Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer,
NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception;
}
동작 방식
ArgumentResolver
의 supportsParameter()
를 호출해서 해당 파라미터를 지원하는지 체크하고, 지원하면 resolveArgument()
를 호출해서 실제 객체를 생성한다. 그리고 이렇게 생성된 객체가 컨트롤러 호출 시 넘어가는 것이다.
그리고 원한다면 여러분이 직접 이 인터페이스를 확장해서 원하는 ArgumentResolver
를 만들 수도 있다. 실제 확장하는 예제는 향후 로그인 처리에서 진행하겠다.
헉..정말? 너무 설렌다
ReturnValueHandler
HandlerMethodReturnValueHandler
를 줄여서 ReturnValueHandler
라 부른다. ArgumentResolver
와 비슷한데, 이것은 응답값을 변환하고 처리한다.
Method라는 단어는 컨트롤러들이 Method단위로 작동해서 붙은 이름.
컨트롤러에서 String으로 뷰 이름을 반환해도 동작하는 이유가 바로 ReturnValueHandler덕분이다. 어떤 조율들이 있는지 살짝 코드로 확인만 해보자.
스프링은 10여개가 넘는 ReturnValueHandler
를 지원한다.
예) ModelAndView
, @ResponseBody
, HttpEntity
, String
참고
가능한 응답 값 목록은 다음 공식 메뉴얼에서 확인할 수 있다.
https://docs.spring.io/spring-framework/docs/current/reference/html/web.html#mvc-ann-return-types
Http 메시지 컨버터 :argumentResolver가 사용하는 것.
HTTP 메시지 컨버터 위치
Argument Resolver 중에서 @RequestBody, HttpEntity 처리해야하는 리졸버인 경우에 한해서 Argument Resolver가 로직을 다 처리하는 것이 아니라 HTTP 메시지 컨버터에게 물어본다. "이거 처리해야 하는데, http 메시지 바디에 있는거 꺼내서 너가 json이나 이런거 처리해줄 수 있어?" 하고 물어 보면 http 메시지 컨버터가 "어, 나 할수있어" 하고 바꿔주는 것.
HTTP 메시지 컨버터는 어디쯤 있을까?
HTTP 메시지 컨버터를 사용하는 @RequestBody
도 컨트롤러가 필요로 하는 파라미터의 값에 사용된다. @ResponseBody
의 경우도 컨트롤러의 반환 값을 이용한다.
요청의 경우 @RequestBody
를 처리하는 ArgumentResolver
가 있고, HttpEntity
를 처리하는 ArgumentResolver
가 있다. 이 ArgumentResolver
들이 HTTP 메시지 컨버터를 사용해서 필요한 객체를 생성하는 것이다.(어떤 종류가 있는지 코드로 살짝 확인해보자.)
응답의 경우 @ResponseBody
와 HttpEntity
를 처리하는 ReturnValueHandler
가 있다. 그리고 여기에서 HTTP 메시지 컨버터를 호출해서 응답 결과를 만든다.
스프링 MVC는 @RequestBody
@ResponseBody
가 있으면 RequestResponseBodyMethodProcessor
(ArgumentResolver) HttpEntity
가있으면 HttpEntityMethodProcessor
(ArgumentResolver)를사용한다.
참고
HttpMessageConverter
를 구현한 클래스를 한번 확인해보자.
@RequestBody가 들어오면, argumentResolver 안의 supportParameter()를 통해 해당 객체가 담당하는 건지 확인해보고, readWithMessageConverters()에서 아까 본 메세지 컨버터들(ex.StringHttpMessageConverter
,MappingJackson2HttpMessageConverter
)을 loop 돌리면서 적절한 것을 골라 사용한다. 해당 HTTP 메시지 컨버터가 read()를 사용해서 메시지를 읽어온다.
스프링은 다음을 모두 인터페이스로 제공한다. 따라서 필요하면 언제든지 기능을 확장할 수 있다.
HandleMethodArgumentResolver
HandleMethodReturnValueHandler
HttpMessageConverter
스프링 필요한 대부분의 기능을 제공하기 때문에 실제 기능을 확장할일이 많지는 않다. 만일 필요한 경우 기능 확장은 WebMvcConfigurer
를 상속받아서 스프링 빈으로 등록하면 된다. 실제 자주 사용하지는 않으니 실제 기능 확장이 필요할 때 WebMvcConfigurer
를 검색해보자.
WebMvcConfigurer 확장
@Bean
public WebMvcConfigurer webMvcConfigurer() {
return new WebMvcConfigurer() {
@Override
public void addArgumentResolvers(List<HandlerMethodArgumentResolver> resolvers) {
//...
}
@Override
public void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
//...
}
};
}