x-www-form-urlencoded에게 (도움을 받아) 승리하다!

박화랑·2025년 3월 18일
1

Spring_6기

목록 보기
14/15

1. x-www-form-urlencoded 방식으로 두 개의 객체를 받으려 했던 시도

서론

  • 강의를 듣던 중 @ModelAttribute@RequestParam을 생략해도 필드명이 같으면 자동으로 바인딩된다는 내용을 접했다.
  • 여기서 "객체가 2개라면 어떻게 될까?" 라는 의문이 생겨 실험을 진행하였다.
  • 튜터(Tutor)튜티(Tutee) 정보를 동시에 받아 컨트롤러에서 처리하려고 했다.
  • HTTP 요청 방식은 application/x-www-form-urlencoded 를 사용했다.

목표

  • 두 개의 객체를 @ModelAttribute를 통해 바인딩하는 것이 가능할지 확인.
  • 각 객체가 자신의 필드에만 데이터를 바인딩해야 함.
  • 객체명이 다르므로(튜터 vs 튜티) Spring이 이를 자동으로 구별할 것이라 예상.

Postman 요청 (x-www-form-urlencoded)

KeyValue
tutor.name김철수
tutor.age26
tutee.name박화랑
tutee.cost10000

컨트롤러 코드

@PostMapping("/v5/tutor")
public String requestParamV5(
        @ModelAttribute Tutor tutor,
        @ModelAttribute Tutee tutee
) {
    return "tutor name = " + tutor.getName() + " age = " + tutor.getAge() +
           " and tutee name = " + tutee.getName() + " cost = " + tutee.getCost();
}

2. 발생한 문제

  • @ModelAttribute요청 파라미터에서 객체명을 기준으로 자동 매핑하지 않음.
  • 즉, tutor.name, tutee.nameSpring이 별도의 객체로 구분하지 않고, 필드명(name, age, cost)만 기준으로 매핑하려고 시도함.
  • 그 결과 TutorTutee 객체의 필드가 충돌하고, null이 들어가는 문제 발생.

❌ 출력 결과 (문제 발생)

tutor name = null age = 0 and tutee name = null cost = null
  • name, age, cost 같은 필드명이 여러 객체에서 중복되었지만, Spring이 이를 제대로 구분하지 못함.
  • 결과적으로 요청 데이터가 객체에 정상적으로 매핑되지 않음.

3. 해결책

시도한 해결책들

✅ 1. @ModelAttribute("tutor")으로 변경

@PostMapping("/v5/tutor")
public String requestParamV5(
        @ModelAttribute("tutor") Tutor tutor,
        @ModelAttribute("tutee") Tutee tutee
) { ... }

결과:

  • @ModelAttribute("tutor")뷰(View)에서 사용할 때만 객체명을 설정하는 용도이며, 요청 데이터를 바인딩할 때는 영향을 주지 않음.
  • 즉, tutor.nameTutor 객체의 name 필드로 자동 매핑하지 않음.

✅ 2. @ModelAttribute + DTO 방식

  • 박**님의 조언을 토대로 다시 시도 후에 성공!!
@PostMapping("/v5/tutor")
public String requestParamV5(@ModelAttribute TutorAndTutee tutorTuteeDTO) {
    Tutor tutor = tutorTuteeDTO.getTutor();
    Tutee tutee = tutorTuteeDTO.getTutee();
    return "tutor name = " + tutor.getName() + " age = " + tutor.getAge() +
           " and tutee name = " + tutee.getName() + " cost = " + tutee.getCost();
}

결과: ✅ 정상 동작!

  • DTO를 사용하면 내부 객체도 바인딩 가능하여 문제가 해결됨.

4. @RequestBody 방식으로 전환한 경우

application/json 방식으로 요청 변경

{
  "tutor": {
    "name": "김철수",
    "age": 26
  },
  "tutee": {
    "name": "박화랑",
    "cost": "10000"
  }
}

컨트롤러 코드

@PostMapping("/v5/tutor")
public String requestParamV5(@RequestBody TutorTuteeDTO request) {
    Tutor tutor = request.getTutor();
    Tutee tutee = request.getTutee();
    return "tutor name = " + tutor.getName() + " age = " + tutor.getAge() +
           " and tutee name = " + tutee.getName() + " cost = " + tutee.getCost();
}

5. @ModelAttribute vs @RequestBody 비교

구분@ModelAttribute + DTO@RequestBody + DTO
요청 방식application/x-www-form-urlencodedapplication/json
내부 객체 지원✅ 가능 (DTO 내부 객체 필드 자동 바인딩)✅ 가능 (JSON -> DTO 변환)
유효성 검사 지원✅ 가능 (@Valid 사용 가능)✅ 가능 (@Valid 사용 가능)
복잡한 데이터 구조❌ 어렵다 (중첩 객체가 깊어지면 복잡함)✅ 유연함 (JSON 구조로 깊이 있는 데이터 처리 가능)
유지보수성✅ (단순한 경우 적합)✅ (복잡한 경우 더 적합)

결론

  • DTO를 사용하면 @ModelAttribute도 가능하지만, 중첩 객체가 많아지면 @RequestBody + JSON 방식이 더 적합함.
  • Spring에서는 @RequestBody를 활용한 JSON 방식을 권장하며, 이를 사용하면 객체를 명확하게 구분할 수 있고, 유효성 검사가 쉬워짐.
  • 즉, @ModelAttribute는 단순한 경우 가능하지만, 데이터가 복잡하면 @RequestBody가 더 적합함!
profile
개발자 희망생

0개의 댓글

관련 채용 정보