-
Notifications
You must be signed in to change notification settings - Fork 134
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[Spring Core] 남해윤 미션 제출합니다. #391
base: haeyoon1
Are you sure you want to change the base?
Conversation
- Gradle 수정 - HomeController, ReservationController 생성 - Reservation dto 생성
- ReservationController속 변수에 값 넣어준 코드 주석처리
- h2 데이터베이스를 활용하여 데이터를 저장하도록 수정
- 예약 조회 API 처리 로직에서 저장된 예약을 조회할 때 데이터베이스를 활용하도록 수정
- 예약 추가/취소 API 처리 로직에서 데이터베이스를 활용하도록 수정
- time 관련 controller, dao, dto, entity, repository, service 추가
-reservation time의 문자형을 Time형으로 수정
- 테이블 스키마 재정의 - 예약 클래스 시간 타입 Time으로 수정 - 예약 쿼리 수정
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
안녕하세요, 해윤님!
미션 하시느라 수고 많으셨습니다.
Repository와 Dao, controller, service로 계층화가 잘 되어있는 것 같아요.
몇가지 코멘트 남겼으니 확인해주세요!
추가로 깨지는 테스트도 같이 수정 하신 후 실행 가능한 상태로 전달주세요!
질문에 대한 답변
원래 조건대로 String형태로 지정하는 것이 나을까요?
이 부분에 대해 고민이 남아있는 코드 같아요.
시간 정보를 String
이 아닌 시간 관련 객체로 변환하는 작업이 묻어 있더군요.
꼭 시간 정보를 LocalTime
, LocalDateTime
으로 저장할 의무는 없습니다. 이것 역시 선택의 영역이에요.
Java에서는 String
대신 LocalDateTime
과 같은 타입을, DB에서는 VARCHAR
대신 TimeStamp
혹은 datetime
과 같은 시간 정보를 위한 타입을 제공 하는데요.
이러한 타입은 시간들 사이에 연산(날짜 추가 등)을 제공하기도 하고, Timezone
으로 세계시간을 반영 할 수도 있고, 해당 정보에서 년도, 달, 일을 추출 하는 등 날짜에 특화된 다양한 기능들을 제공합니다.
이런 기능이 필요하다면 직접 만들어서 사용 하기 보단 해당 타입을 사용 하는게 낫겠죠.
그리고 제 경험상, 시간 정보를 다루는 경우에는 거의 모든 상황에서 해당 타입을 사용하는게 String
타입 을 유지하는 것보다 유용합니다.
해당 타입들이 어떤 기능들을 제공하는지 찾아보시고, 익숙해지면 매우 도움 될겁니다!
그래서 저는 시간 정보에 대해 String
타입보단 LocalDateTime
이나 LocalTime
과 같은 타입으로 사용하는 것을 권장드려요!
그러면, 어느 계층에서 해당 타입을 사용할 지, 언제 변환할 지 고민 하셔야 합니다.
저는 ui 계층에선 String
, presentation -> application
으로 전달할 때 변환 할 것 같아요.
반대로 응답을 만드는 상황에서도 application -> presentation
로 응답할 때 변환하고요.
이유는, 아시다시피 날짜/시간 형식이 너무나도 다양하게 있습니다. ex) "07:09", "7:09", "07-09"
presentation` 영역에서는 각 클라이언트 마다 형식을 받고, application에서 사용 될 수 있도록 변환하는 거죠.
"07:09", "7:09", "07-09"
처럼 다양한 형식에 대해서도 application
에서 정의한 time
객체로만 변환 하여 전달 한다면 형식에 상관없이 로직을 수행 할 수 있죠.
그리고, database에 저장할 때(persistence) 역시 DB에 정의 된 타입으로 전달 할 수 있습니다.
DB에서 VARCHAR
를 사용하고 있다면, Time -> String 으로의 변환이 필요할 테고,
아니면 DB에서 TimeStamp 타입으로 정의하면 그대로 저장하면 됩니다!
jdbcTemplate에서 timestamp로 저장하는 방식을 간단하게 정리한 글 공유드릴게요. 참고하시면 도움 될 것 같습니다.
조금 작업이 들 수 있지만, 시간 데이터에 대해서 String
대신 제공 되는 시간 관련 객체를 사용하시면 학습에도, 앞으로의 개발에도 도움이 될 것 같습니다!
@Controller | ||
public class TimeController { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Controller는 전체적으로 깔끔하네요!👍👍
String query = "SELECT id FROM reservation ORDER BY id DESC LIMIT 1"; | ||
Long id = jdbcTemplate.queryForObject(query, Long.class); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
방금 저장된 저장한 예약의 id를 조회하는 것 같아요.
이 방식은 여러 저장이 이루어지는 환경에서 문제가 생길 수 있어요.
저장과 동시에 DB에서 생성된 id를 반환하는 방식을 활용해보세요!
jdbcTemplate
에서는 KeyHolder를 사용해볼 수 있어요
String sql = "INSERT INTO time(time) VALUES (?)"; | ||
jdbcTemplate.update(sql, time.getTime()); | ||
|
||
String query = "SELECT id FROM time ORDER BY id DESC LIMIT 1"; | ||
Long id = jdbcTemplate.queryForObject(query, Long.class); | ||
|
||
return new Time(id, time.getTime()); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
방금 저장한 시간의 id를 조회하는 것 같아요.
이 방식은 여러 저장이 이루어지는 환경에서 문제가 생길 수 있습니다.
27번 라인 이후 30번 라인의 실행 되지 전에 다른 쓰레드에서 저장이 이루어졌다면...?
저장과 동시에 DB에서 생성된 id를 반환하는 방식을 활용해보세요!
jdbcTemplate
에서는 KeyHolder를 사용해볼 수 있어요
return time; | ||
} | ||
|
||
public LocalTime getTimeASALocalTime(){ //string->LocalTime |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
메서드 네이밍 컨벤션에 맞춰주세요!
public LocalTime getTimeASALocalTime(){ //string->LocalTime | |
public LocalTime getTimeAsLocalTime(){ //string->LocalTime |
Reservation reservation = new Reservation( | ||
requestDto.getName(), | ||
requestDto.getDate(), | ||
requestDto.getTime() | ||
requestDto.getStringTimeAsTime() | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Dto -> Entity로의 변환 작업을 dto에 편의 메서드를 추가해볼 수도 있을 것 같아요.
class ReservationRequestDto {
// ...
public Reservation toReservation() {
// 변환 로직
}
}
Dto 객체에 변환 로직을 추가하여 getStringTimeAsTime()
메서드를 굳이 제공하지 않아도 될거 같네요!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ReservationService 클래스의 createReservation 메서드에
Long timeId = parseTimeId(requestDto.getTime());
Time time = findTimeById(timeId);
Reservation reservation = new Reservation(
requestDto.getName(),
requestDto.getDate(),
time
);
이렇게 수정했는데 처음에 코멘트 남겨주신대로 dto에 변환 로직을 만들어서 작성하는 것이 나을까요?
.map(time -> new TimeResponseDto( | ||
time.getId(), | ||
time.getTimeASALocalTime())) | ||
.collect(Collectors.toList()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
아래처럼 변경할 수 있어요.
.collect(Collectors.toList()); | |
.toList(); |
약간의 차이점이 있으니 알고계시면 좋습니다!
src/main/resources/schema.sql
Outdated
id BIGINT NOT NULL AUTO_INCREMENT, | ||
name VARCHAR(255) NOT NULL, | ||
date VARCHAR(255) NOT NULL, | ||
time_id BIGINT, // TODO: 수정 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sql의 주석은 --
로 표기해요.
잘못된 주석 표기로 Sql 문법 오류가 나서 테스트 실행이 안되고 있어요ㅠ
time_id BIGINT, // TODO: 수정 | |
time_id BIGINT, -- TODO: 수정 |
# Conflicts: # src/main/java/roomescape/dao/ReservationDao.java # src/main/java/roomescape/dto/ReservationRequestDto.java # src/main/java/roomescape/entity/Reservation.java # src/main/java/roomescape/entity/Time.java # src/main/java/roomescape/repository/ReservationRepository.java # src/main/java/roomescape/service/ReservationService.java # src/main/java/roomescape/service/TimeService.java
551b7a8
to
6d9acfd
Compare
리뷰어님 피드백 받아 수정하였습니다! 혜빈님이 Reservation 엔티티에 Time 객체가 Localtime을 import하고있지 제가 만든 time 클래스를 가르키고 있지 않은 것 같다고 하셔서 해당 방법대로 수정하고 많이 시도 해보았으나 ... 여전히 아직 9단계는 해결하지 못하였습니다 ㅠㅠㅠㅠ 혹시 수정이 필요한 부분있다면 말씀해주세요! 또한 틀린부분 이외에도 다양한 피드백 부탁드립니다! 감사합니다:) |
리뷰 반영 확인했습니다! 전체적으로 발생하는 오류는 private Long parseTimeId(String time) { // String -> long
try {
return Long.parseLong(time);
} catch (DateTimeException e) {
throw new IllegalArgumentException("Invalid time id format: " + time);
}
} 시간 요청 값으로 "11:00"이 주어지는데, 이걸 바로 아마 아래처럼 저장된 DB 테이블에서 해당 ID를 찾는 것 같아요.
시간 "11:00"에 대한 Id는 1이고, 이 값 찾아 예약을 저장을 하는 로직인 것 같습니다. 요청으로 들어온 "11:00"을 잘 처리하면 해결 할 수 있을 것 같아요! |
안녕하세요 원태연 리뷰어님! 리뷰 잘 부탁 드립니다!
지난주에 다른 리뷰어님께서 time의 자료형을 String이 아니라 LocalTime으로 변경해보라는 리뷰를 남겨주셔서 그렇게 변경해 보았는데 이번주 미션에서 오류가 너무 많이 나더라고요 .. 원래 조건대로 String형태로 지정하는 것이 나을까요?
그리고 최대한 코드를 수정해보았는데 9단계 미션부터 오류가 나서 실행되지 않습니다.... 추가로 더 수정해서 다시 pr올리도록 하겠습니다 ㅠㅠ 또한 어느부분이 잘못됐는지 궁금합니다. 늦게 제출하여 죄송합니다. 리뷰 잘 부탁드립니다! 감사합니다.