BackEnd_Servers

@RequestBody, @ModelAttribute, @RequestParam

wookjae 2021. 10. 8. 11:01

1. RequestBody, ModelAttribute, RequestParam 설명

 

@RequestParam

@RequestParam은 1개의 HTTP 요청 파라미터를 받기 위해서 사용한다. @RequestParam은 필수 여부가 true이기 때문에 기본적으로 반드시 해당 파라미터가 전송되어야 한다. 해당 파라미터가 전송되지 않으면 400 Error를 유발하게 된다. 그렇기 때문에 반드시 필요한 변수가 아니라면  required의 속성 값을 false로 설정해둘 수 있으며 해당 Parameter를 사용하지 않고 요청을 보낼 경우에 default로 받을 값을 defaultValue 속성을 지정하여 설정할 수 있다.

 

@RequestBody ? 

@RequestBody는 클라이언트가 전송하는 JSON(application/json) 형태의 HTTP Body 내용을 Java Object로 변환시켜주는 역할을 한다. 그렇기 때문에 Body가 존재하지 않는 Get 메소드에 @RequestBody를 활용하려고 한다면 에러가 발생하게 된다. @RequestBody로 받는 데이터는 Spring에서 관리하는 MessageConverter들 중 하나인 MappingJackson2HttpMessageConverte를 통해 Java 객체로 변환된다.

Spring은 메세지를 변환되는 과정에서 객체의 기본 생성자를 통해 객체를 생성하고, Reflection을 사용해 값을 할당하고, 이러한 이유로 @RequestBody에는 Setter가 필요 없다.

 

@ModelAttribute ? 

@ModelAttribute는 클라이언트가 전송하는 (multipart/form-data 형태의 HTTP Body 내용, HTTP 파라미터들을 Java Object(DTO)의 Setter를 통해 1:1로 Binding)하기 위해 사용된다. @ModelAttribute에는 매핑시키는 파라미터의 타입이 객체의 타입과 일치하는지를 포함한 다양한 검증(Validiation) 작업이 추가적으로 진행된다. 예를 들어 게시물의 번호를 저장하는 int형 index 변수에 "첫번째" 라는 String형을 넣으려고 한다면, BindException이 발생하게 된다. 즉, JSON이나 XML과 같은 형태의 데이터를 MessageConverter를 통해 변환시는 @RequestBody와 달리, @ModelAttribute는 multipart/form-data 형태의 HTTP Body와 HTTP 파라미터들을 매핑시킨다는 차이가 있다.

 

극단적으로 @ModelAttribute와 @RequestBody를 비교하자면, @ModelAttribute는 바인딩시키는 어떤 데이터를 set해주는 Setter함수가 없다면 매핑이 되지 않는다. 하지만 @RequestBody는 요청받은 데이터를 변환시키는 것이기 때문에, Setter함수가 없어도 값이 매핑이 된다.

 

추가로, @ModelAttribute 어노테이션을 활용해서 특정 Parameter만을 받아올 수도 있다.

예를들어 { writer: 'jungukjae', contents : 'springexercise' }의 JSON형태로 데이터를 전송했다고 하면, 컨트롤러에서는@ModelAttribute('writer') String writer의 형태를 활용하여 writer 변수에 'jungukjae' 만을 Binding하여 사용가능 하다.




2. RequestBody, ModelAttribute, RequestParam 실습 

 

[ Model 객체 ]


@Getter
 
@Setter 
@ToString 
public class Board { 

    private int index; // 게시물의 번호

    private String writer; // 글쓴이 
 
    private
 String contents; // 게시물 내용

}

 

[ Controller ]

@RestController 
@RequestMapping("/board") 
@RequiredArgsConstructor 
@Log4j2 
public class BoardController { 
   
    private
 final BoardService boardService; 

    @PostMapping("/requestBody") 
    public ResponseEntity<Board> requestBodyMethodTest(@RequestBody final Board board) { 

        // @RequestBody는 JSON 형태의 HTTP Body를 MessageConverter를 통해 Java 객체로 변환시킨다. 
        log.info(board); 
       
        return
ResponseEntity.ok(boardService.add(board));
    }

    @PostMapping("/modelAttribute") 
    public ResponseEntity<Board> modelAttributeMethodTest(@ModelAttribute final Board boad) { 

        // @ModelAttribute는 multipart/form-data 형태의 HTTP Body 내용과 HTTP 파라미터들을 1:1객체바인딩. 
        // 만약, Setter함수가 없다면 매핑을 시키지 못하고, Null이 지정된다.
 
        log.info(board);
 
        return ResponseEntity.ok(boardService.add(board));
    }
 
    @GetMapping("/list") 
    public ResponseEntity<List<Board>> requestParamMethodTest( 
           @RequestParam(value = "searchKeyWord1", required = false) final String search1, 
           @RequestParam(value = "writer", defaultValue = "jungukjae") final String search2 ) { 
       
        // 파라미터 search1은 required가 false이기 때문에 없을 수도 있다.
 
        // 파라미터 search2는 required가 true이기 때문에 반드시 요청 파라미터로 존재해야 한다. 

        final List<Board> boardList = search1 != null 
                               ? boardService.getBoardList(search1)
                               : boardService.getBoardList();    // DB로 부터 게시판 목록조회.. 



        log.info(search2); 

        return
 ResponseEntity.ok(boardList);
    }
}



 

위 설명한 내용에 따르면, ModelAttribute는 multipart/form-data형태의 HTTP Body내용과 HTTP 파라미터(Query Param)들을 Java Object(DTO)의 Setter를 통해 1:1로 Binding 된다고 하였다. 

 

-- 아래, CASE(1,2) Server에서 정상적으로 Binding.. 


CASE1. (HTTP-GET)

Request URL - QueryParam
http://localhost:8080/board/modelAttribute?contents=springexercise


CASE2. (HTTP-POST / type - multipart/form-data) 
Request BODY 
{
    index :1 
  , writer : "jungukjae" 


 


3. 설명요약

 

  • RequetParam
    • 1개의 HTTP 파라미터를 얻기 위해 사용된다.
    • 필수 여부가 true이기 때문에 반드시 필요한 경우가 아니라면 required=false 옵션을 설정해야 한다.
  • RequestBody
    • JSON으로 받은 HTTP Body 데이터를 MessageConverter를 통해 변환시킨다.
    • 바인딩이 아닌 변환을 시키므로(Reflection), 변수들의 Setter함수가 없어도 정상적으로 변수들이 저장 및 활용가능하다. 
  • ModelAttribute
    • multipart/form-data 형태의 HTTP Body 내용과 HTTP 파라미터들을(QuerParam - HttpGET) Java Object(DTO)의 Setter를 통해 1:1로 Binding 된다.
    • 변환이 아닌 바인딩을 시키므로, 변수들의 Setter함수가 없으면 변수들이 저장되지 않는다. (Setter함수가 없을 시.. Null로 셋팅된다)

 

 

exit..