전통적인 Spring MVC의 컨트롤러인 @Controller는 주로 View를 반환하기 위해 사용합니다.
아래와 같은 과정을 통해 Spring MVC Controller는 Client의 요청으로부터 View를 반환합니다.
@Controller가 View를 반환하기 위해서는 ViewResolver가 사용되며 ViewResolver 설정에 맞게 View를 찾아 랜더링합니다.
Spring MVC의 컨트롤러에서도 Data를 반환해야 하는 경우도 있습니다.
Spring MVC의 컨트롤러에서는 데이터를 반환하기 위해 @ResponseBody 어노테이션을 활용해주어야 합니다. 이를 통해 Controller도 Json 형태로 데이터를 반환할 수 있습니다.
@RestController가 Data를 반환하기 위해서는 viewResolver 대신에 HttpMessageConverter가 동작합니다.
HttpMessageConverter에는 여러 Converter가 등록되어 있고 반환해야 하는 데이터에 따라 사용되는 Converter가 달라집니다. 단순 문자열인 경우에는 StringHttpMessageConverter가 사용되고, 객체인 경우 MappingJackson2HttpMessageConverter가 사용되며 데이터 종류에 따라 서로 다른 MessageConverter가 작동하게 됩니다.
Spring은 클라이언트의 HTTP Accept 헤더와 서버의 컨트롤러 반환 타입 정보 둘을 조합해 적합한 HttpMessageConverter를 선택하여 이를 처리합니다.
@Controller
@RequestMapping("/user")
@RequiredArgsConstructor
public class UserController{
private final UserService userService;
@PostMapping("/info")
public @ResponseBody User info(@RequestBody User user){
return userService.userInfo(user);
}
@GetMapping("info/view")
public String infoView(Model model, @RequestParam(value="userName", required=true) String userName){
User user = userService.userInfo(userName);
model.addAttribute("user", user);
return "/user/userInfoView";
}
}
위 예제는 info라는 User라는 데이터를 반환하고자 하고 있고 User를 json으로 반환하기 위해 @ResponseBody라는 어노테이션을 붙여주고 있습니다. infoView 함수에서는 View를 전달해주고 있기 때문에 String을 반환값으로 설정해주었습니다.
@RestController는 Spring MVC Controller에 @ResponseBody가 추가된 것입니다.
@RestController의 주용도는 Json 형태로 객체 데이터를 반환하는 것입니다.
@RestController
@RequestMapping("user")
@RequiredArgsConstructor
public class UserController{
private final UserService userService;
@PostMapping("info")
public ResponseEntity<User> info1(@RequestBody User user){
return ResponseEntity.ok(userService.userInfo(user));
}
@PostMapping("info2")
public ResponseEntity<User> info2(@RequestParam(value="userName", required=true) String userName){
User user = userService.userInfo(userName);
if(user == null){
return ResponseEntity.noContent().build();
}
return ResponseEntity.ok(user);
}
@PostMapping("info3")
public ResponseEntity<User> info3(@RequestParam(value="userName", required=true) String userName){
return Optional.ofNullable(userService.userInfo(userName)
.map(user -> ResponseEntity.ok(user))
.oreElse(ResponseEntity.noContent().build());
}
}
info1의 메소드는 User 데이터를 그대로 반환하고 있습니다.
하지만 이렇게 처리하는 것 보다 info2처럼 결과 데이터와 상태코드를 함께 제어하여 반환하는 것이 좋습니다.
또한 만약 userService에서 반환하는 형태가 Optional이라면 info3 처럼 깔끔하게 처리를 해줄 수 있습니다.