이번주는 개발 마지막 주라서 마무리를 했다.

 

알고리즘

역시나 브론즈만 골라서 풀었다.

 

프로젝트 개발 마무리 및 발표

저번 주말부터 QA를 하며 개발 마무리를 하였고, 수요일부터 제출을 위한 산출물 작성을 하였다.

그리고 금요일 발표를 하였다. 발표 + UCC + 시연 으로 진행되었고, 나는 발표자가 아니었지만 우리 팀원들 너무 고생하셨습니다 ㅠ

 

후기

프로젝트 하나가 끝났다.

우리팀이 팀워크 부분에서는 최고라고 생각하지만 나는 다른 프로젝트를 위해 팀을 이탈하였고, 취업을 해서 이탈한 팀원이 2명 있었다.

그리고 팀원들 간에 트랙도 나뉘게 되어 사실상 팀이 분해가 되었다.

벌써 끝이라는 사실이 믿기지가 않았고, 팀원 6명이 모두 흩어질 것이라고는 더 믿기지 않았다..

능력없는 팀장인 저를 이끌어주신 팀원분들 너무 감사드리고 고생하셨고, 취업하신 분들은 너무 축하드려요!!

3차 프로젝트때 다시 모였으면 좋겠다는 생각이 들었다.

이번주는 프로젝트 마무리 이후에 오랜만에 휴식을 취했다.

원래 쉬면 안된다는 생각이 지배적이기에 걱정이 들고있지만 다음에 쉬지않기 위해 몰아서 쉰다는 생각을 하자.

다음주부터 있을 프로젝트는 처음보는 것들이라 모든 부분에서 더 힘들거기 때문에 악착같이 따라가야겠다는 생각이 들었다.

'잡담' 카테고리의 다른 글

8월 4주차 결산  (0) 2021.08.29
900  (0) 2021.08.24
8월 2주차 결산  (0) 2021.08.15
8월 1주차 결산  (0) 2021.08.08
7월 5주차 결산  (2) 2021.08.01

이번주는 사실상 개발 마지막 주라는 생각으로 거의 개발만 진행하였다.

 

알고리즘

이번주도 역시 잔디 채우기 실천중이다.

이번주도 브론즈만 골라서 풀었고, 가끔 개발도중 의도한대로 안되고 화가나서 플래문제도 한문제 풀었었다 ㅎ..

다음 프로젝트는 더 힘들건데 아마 더 많이 풀지않을까 하는 예상을 해본다.

 

프로젝트 개발 진행

저번주와 마찬가지로 이번주차에 할 일들을 지라에 등록하였다.

1000개에 가까워져가는 이슈 수를 보면서 우리 팀원들이 자랑스러웠다.

이번주는 팀원들과 배포를 진행하고, 작성한 Servive, Repository에 대한 Test를 다시 작성하였다.

백엔드가 일찍 마무리되었고 프론트엔드와 맞춰가며 수정해야하는 단계에 도달하였기에 우리팀 전원이 프론트엔드에 투입되었다.

프론트.. 너무어렵다. 하지만 도움이 되었다니 정말 다행이다.

금요일 팀장미팅 시간에 다른 팀의 진행상황들을 보았지만 우리팀 정말 대단하다는 생각이 들었다.

 

버그 수정

프로젝트를 진행하면서 백엔드에서 잘못 구현된 로직이 생각보다 많이 있었다.

Pagenation 이후 Sort가 대표적인 예시다.

그 외 자잘한 버그들은 수정을 완료했으며, Pagenation Sort부분은 블로그 포스팅을 완료하였다.

 

File업로드 학습

저번주에도 작성했었던 내가맡지 않은 File업로드 부분이다.

File과 Dto를 Controller에서 같이 받을 수 있는 방법을 계속 찾았었고, 팀원분이 RequestPart라는걸 알려주셔서 연습을 해보았다.

그 과정에서 Postman으로 Post 요청을 보내보았고, 400, 415에러들을 마주한 것을 해결하였다.

이 과정 역시 블로그에 포스팅을 완료하였다.

 

소스콘 프로젝트 팀 빌딩

싸피 우수교육생을 선발하여 교육과정과는 조금 다른 프로젝트를 진행한다고 한다.

나는 여기에 선발되었고, 팀 빌딩과 트랙 선정이 완료되었다.

처음 해보는 것이기도 해서 걱정이 많이 되지만 1학기때와 같은 행동은 하지 않을 것이다.

 

후기

그냥 개발하느라 정신없던 한주였다.

블로그 포스팅도 주중에 할려고 했지만 하지 못했고, 주말에서야 겨우 완료하였다.

개발을 진행하면서 별 못보던 에러까지 다 본것같다.. 왜이렇게 많죠?

하지만 이런 에러들을 마주하면서도 나의 경험치는 쌓이고 있다는 생각이 들기 때문에 그냥 열심히 해야겠다.

이번 프로젝트의 마지막 주말이라서 그런지 팀원들이 주말인데도 자발적으로 디코, 노션을 이용하여 개발에 참여하고있다.

아직 한주 남았지만 정말 고생했다고 말해주고싶고, 특화 프로젝트때는 떨어지겠지만 자율 프로젝트때는 꼭 다시 만났으면 좋겠다 ㅠ

남은 시간 화이팅!

'잡담' 카테고리의 다른 글

900  (0) 2021.08.24
8월 3주차 결산  (0) 2021.08.22
8월 1주차 결산  (0) 2021.08.08
7월 5주차 결산  (2) 2021.08.01
solved.ac에 잔디가 생겼어요!  (0) 2021.07.27

보통 Controller에서 Dto를 받을 때는 @RequestBody를 주로 사용합니다.

그리고 File을 받을 때는 MultipartFile 객체를 사용하며, @RequestParam을 사용합니다.

 

하지만 File과 Dto를 같이 받기 위해서는 @RequestPart라는 어노테이션이 필요합니다.

File은 @RequestParam, Dto는 @RequestBody를 사용해서 받을려고 했지만 몇번을 시도해도 안되더라고요..

그래서 구글링을 통해 RequestPart라는 어노테이션을 찾아냈습니다.

 

우선 Controller입니다.

File과 Dto 모두 @RequestPart 어노테이션으로 받도록 합니다.

 

그리고 Dto인 UserInsertPostReq는 이렇게 구성되어 있습니다.

이전글 에서 사용했던 Dto와 동일합니다.

 

그리고 Controller에서 호출할 Service의 insertUser 메소드입니다.

파일을 저장할 경로를 지정한 후에 지정한 경로에 파일을 저장하고 DB에 유저 정보를 추가하는 로직입니다.

 

저는 이를 수행하기 위해 포스트맨을 사용하였습니다.

Key name은 @RequestPart 에서 value로 지정해준 name과 동일한 Key name으로 넣어줍니다.

이제 저기있는 Send를 눌러볼까요?

 

네 역시나 400에러가 발생합니다.

 

구글링을 통해 value에 json형식으로 넣는 방법을 알게되었고, 그대로 따라해보았습니다.

이런식으로 Key name은 userInsertPostReq, value는 json 방식으로 작성하였습니다.

다시 Send를 눌러보겠습니다.

 

이번에는 415 에러가 발생합니다..

 

한 두 시간 동안은 400, 415에러만 봤던것 같습니다.

 

이 에러들을 해결하기 위해 구글링을 해본 결과로는

여기서 Dto인 userInsertPostReq는 content type을 application/json으로 지정하면 해결할 수 있습니다.

userInsertPostReq의 content type을 application/json으로 지정한 후 Send를 누릅니다.

 

그럼 입력한 정보가 잘 들어가는 것을 확인할 수 있습니다.

 

File과 Dto를 같이 받아올 때 Key name과 content type 설정을 잘못하여 나온 에러 상황이었습니다.

이 방법 외에 다른 방법이 있는지는 잘 모르겠습니다. 더 많은 공부가 필요할 것 같습니다.

 

지금까지 단일 File과 Dto를 요청하는 것에 대해 알아보았고, 다수의 File과 Dto 요청에 관해서는 아래 링크에서 확인하시면 됩니다.

https://emoney96.tistory.com/375

 

Postman을 이용한 다수의 File, Dto 동시 Post 요청

https://emoney96.tistory.com/258 Postman을 이용한 MultipartFile, Dto 동시 Post요청 보통 Controller에서 Dto를 받을 때는 @RequestBody를 주로 사용합니다. 그리고 File을 받을 때는 MultipartFile 객체를 사..

emoney96.tistory.com

 

프로젝트 진행 중 JpaRepository에서 Pagenation을 적용한 Page객체를 받은 이후 Collections.sort로 정렬을 하였습니다.

하지만 이와 관련된 문제점 하나를 발견하게 되었고 포스팅으로 남길려고 합니다.

 

우선 db에 유저의 정보를 저장해줍니다.

profile_url은 file저장을 공부하면서 사용해본 것이며 이 글과는 상관 없는 file입니다.

내일쯤 file 저장관련 글을 포스팅할 예정입니다.

 

각 유저들의 점수(score)를 같이 저장해줍니다.

이 유저들을 score기준 내림차순으로 조회를 하는데 Pagenation을 적용시키려고 합니다.

 

 

UserRepository에서는 Page 객체로 리턴하는 findAll 메소드를 작성합니다.

 

UserServiceImpl에서 UserRepository의 findAll 메소드를 호출하는 findAllUser 메소드를 작성합니다.

userRepository.findAll로 Page객체를 가져오고, UserGetRes의 of 메소드에서 정렬을 진행합니다.

 

Pagenation을 적용했기 때문에 변수를 많이 사용하였지만 of 메소드의 정렬하는 부분이 핵심입니다.

Page객체를 가져와서 score기준 내림차순으로 정렬을 진행합니다.

 

그리고 Controller에서는 page와 size를 RequestParam으로 받아서 UserService의 findAllUser 메소드를 호출합니다.

 

이제 스웨거를 통해 테스트를 합니다.

확인할 페이지는 0페이지, 즉 첫번째 페이지를 조회하며 최대 3개의 데이터를 조회하려고 합니다.

 

결과는...

네.. 이렇게 나왔습니다.

 

원래 의도한 대로라면 score가 120인 id10부터 score가 100인 id2까지 나오는게 맞습니다.

여기서 확인해봐도 id는 10 -> 5 -> 2 순서대로 조회가 되어야합니다.

 

이 과정에서 제가 치명적인 실수를 한 것이 있습니다.

바로 Repository에서 Page객체를 가져온 이후에 정렬을 진행했다는 것입니다.

 

Page객체에 있는 데이터는 조회 가능한 모든 리스트가 있는것이 아니라 size만큼의 데이터만 있는 것이었습니다.

따라서 제가 Service에서 Repository를 호출할 때 Sort 기준을 아무것도 주지 않았기 때문에 먼저 추가된 데이터부터 size만큼 데이터가 조회된 것이었습니다.

 

그럼 이제 알맞게 수정을 해야겠죠?

우선 UserRepository의 findAll 메소드에 OrderByScore를 추가하거나 Pageable에 Sort를 추가합니다.

그 후에 UserGetRes의 정렬하는 부분을 지웁니다.

 

일단 UserGetRes의 of 메소드의 정렬부분을 지웁니다.

 

 

먼저 UserRepository의 findAll 메소드 수정하는 방법입니다.

findAllUsers로는 자동완성이 안뜨길래 findUsersByOrderByScoreDesc로 수정합니다.

뒤에 Desc를 안붙이면 default로 Asc가 적용됩니다.

 

 

다음은 UserService의 PageRequest에 Sort를 추가하는 방법입니다.

UserRepository에 있는 findAll 메소드는 그대로 두고 Sort를 추가합니다.

score 기준 DESC를 추가합니다.

 

 

둘 중 아무방법이나 사용한 다음 스웨거를 통해 테스트를 합니다.

그러면 이렇게 올바르게 조회를 할 수가 있습니다.

 

저는 두 번째 방법을 선호하기는 합니다만 두 방법 중 어떤 방법이 더 좋을지는 잘 모르겠습니다.

앞으로 로직을 구현할 때 생각이라는 것을 많이 해야겠다는 생각이 듭니다.

 

정렬을 적용하는 시점이 잘못되어 생긴 문제였습니다.

이번주에는 프로젝트 진행과 관련된 개념공부를 하였다.

 

알고리즘

solve.ac에 잔디를 채워넣기 위해 1일 1백준을 실천중이다.

이번에도 역시 브론즈급 문제만 골라서 10분만 투자하였다.

하면서 알게된 점은 solve.ac 기준 하루의 시작은 오전 6시라는 점이다..

12시가 넘어서 한문제 풀었더니 전날 2개 한걸로 표시가 되더라..

 

프로젝트 개발 진행

저번주까지 기획단계를 진행하였고, 개발을 위한 GitLab, Jira 등록 후 개발을 진행하였다.

백엔드가 사람이 많아서 그런지 생각보다 빠르게 진행되었다.

하지만 내가 맡지 않은 부분들에 대한 이해도 필요하다고 생각한다.

 

더미데이터 작성

팀원들과 더미데이터 작성을 같이 진행하였다.

Faker라는 객체를 발견하였고, 사람의 이름과 이메일, 전화번호 등을 랜덤으로 반환을 해준다는 것에 신기했다.

더미데이터 작성 과정중에 JpaRepository save 관련 문제가 있었고, 해결 과정을 블로그에 포스팅하였다.

 

File업로드 학습

내가 맡지 않은 부분 중 하나가 File 업로드이다.

File 관련 URUD 연습으로 어느정도 이해할 수가 있었다.

File 업로드 연습을 하다가 GeneratedValue 어노테이션을 IdClass와 EmbeddedId 어노테이션과 함께 사용하지 못한다는 점을 알게되었고, 블로그 포스팅까지 완료하였다.

 

JWT 개념 학습

말 그대로 JWT 개념만 학습한 상태.

코드 적용이라던지 그부분을 이번주 남은 시간을 이용해서 진행해야한다.

 

후기

우선 1주차 개발을 빠르게 진행한것 같아서 기분이 좋지만 아직 많이 부족하다는 생각이 들었다.

다음주는 팀미팅에 참여를 못하는 일정도 있다보니 그 공백을 채우기 위해 빡세게 해야겠다.

'잡담' 카테고리의 다른 글

8월 3주차 결산  (0) 2021.08.22
8월 2주차 결산  (0) 2021.08.15
7월 5주차 결산  (2) 2021.08.01
solved.ac에 잔디가 생겼어요!  (0) 2021.07.27
스코페 2차 결과  (0) 2021.04.02

더미데이터를 작성하던 도중 겪었던 문제를 포스팅 하려고합니다.

 

우선 save는 JpaRepository에 내장되어있는 메소드입니다.

save가 수행하는 작업은 insert와 update 이렇게 두가지입니다.

insert와 update로 나뉘는 기준은 select를 했을 때 PK가 있으면 update, 없으면 insert입니다. (물론 null이어도 insert입니다.)

 

우선 User Entity를 작성합니다.

PK는 id이고 auto_increment 속성을 추가하였습니다.

 

그리고 UserRepository를 생성만 해줍니다.

어차피 JpaRepository 내장메소드인 save를 사용하므로 내용은 없습니다.

 

그리고 UserRepository에 대한 Test를 작성해주고 실행합니다.

 

어? save를 두번했으니 2L아닌가요??

심지어는 두번째로 save한 데이터로 덮어져 있습니다.

 

sql문 실행 로그를 확인해보았더니 insert를 했다가 update를 합니다.

 

사실 확인을 위해 save이후의 id를 확인해보았습니다.

 

여기서 insert 직후 id가 1로 출력이 되었습니다.

save(user)를 하였는데 user에 자동으로 insert된 row가 저장되는 모양입니다.

 

이를 해결하기 위해서는 두 번째 save를 하기 전에 id를 null로 바꿔주어야합니다.

user.setId(null);을 추가합니다.

 

이제 insert가 두 번 되었음을 확인할 수 있습니다.

 

db에도 잘 들어갔네요!

 

save를 하면 Entity 객체에 자동으로 값이 들어간다는 사실을 몰라서 겪었던 상황이었습니다.

@GeneratedValue는 @Id와 함께 쓰는 어노테이션이며 auto_increment와 같은 자동 생성 전략을 결정합니다.

 

JPA에서 복합키를 구현하기 위한 IdClass와 EmbeddedId를 복습하던 도중 @GeneratedValue도 함께 사용해보았습니다.

 

IdClass 사용

우선 Primary Key를 위한 UserId 클래스를 생성합니다.

 

그리고 User Entity를 생성합니다.

클래스선언 위에 @IdClass를 사용해줍니다.

 

이제 DB가 만들어지는지 실행을 해볼까요?

음~ 잘 만들어졌네요!

 

이제 User table에 데이터를 넣어봅시다.

데이터를 넣기 위해 Test를 작성하였습니다.

User 테이블은 auto_increment이므로 id는 1이됩니다.

 

하지만 에러가 떠버립니다.

 

 

EmbeddedId 사용

다음은 Embeddable 입니다.

우선 Primary Key를 위한 UserId 클래스를 생성합니다.

 

그리고 User Entity입니다.

PK 위에 @EmbeddedId를 사용합니다.

 

이번에는 auto_increment조차 생기지 않습니다.

다른 방법들도 시도해보았으나 모두 실패했습니다 ㅠ

 

 

결론

사실 제가 코드를 잘못 짠줄 알고 구글링을 여러번 시도했습니다.

https://infondgndg91.blogspot.com/2019/11/hibernate-composite-identifiers.html

 

Hibernate - Composite Identifiers @EmbeddedId @IdClass With @GeneratedValue is impossible

ndgndg91

infondgndg91.blogspot.com

그러던 중 이 블로그를 발견하였고, 이분도 어떤 커뮤니티에서 물어보셨던 것인지

@GeneratedValue를 @IdClass, @EmbeddedId와 함께 사용하는 것이 불가능하다는 답변을 받은것 같습니다.

 

앞으로 주의해서 사용해야 할 것 같습니다.

알고리즘 문제를 풀다보면 "소수점 2자리까지 출력하시오" 라고 하는 문제를 자주 보실겁니다..

이런 문제들을 매번 볼때마다 검색하는 저를 발견하였기에 정리합니다.

 

cout << fixed; 으로 소수점 자릿수를 고정합니다.

그 다음 cout.precision(n);으로 고정할 소수점 자리를 지정합니다.

이제 출력하시면 됩니다.

 

소수점을 3자리로 고정했으니 3자리까지만 출력됩니다.

 

 

다만 위의 사진처럼 소수점 아래에서 반올림을 해서 출력을 하니 이부분 조심해서 사용해야합니다.

'Etc' 카테고리의 다른 글

TypeScript 설치  (0) 2021.08.30
Windows 10 에서 WSL을 이용한 우분투 설치  (0) 2021.08.29
Java 문자열이 정수인지 확인  (0) 2021.06.01
localStorage를 이용한 데이터 저장  (0) 2021.05.11
Java 소수점 자리출력  (0) 2021.03.03

이번주에는 알고리즘, DB 설계, Entity 복합 키 구현, 스켈레톤 코드 작성, 기능 단위 분리를 진행하였다.

 

알고리즘

solved.ac에 추가된 잔디를 채워나가기 위해 1일 1백준을 실천하였고 현재 잔디의 상태는 이러하다.

이전에 했던 문제들은 추가가 안되는 모양이다.

물론 웹에 집중을 하고있기 때문에 브론즈 수준의 문제만 골라서 해결하였다..

 

DB 설계

기능명세, 와이어프레임을 참고하여 진행하였고, 초안 완성 후 컨설턴트님과 코치님의 피드백을 받으며 수정 단계를 거친 후에 마무리를 하였다.

개발 진행을 하면서 기획이 많이 바뀐다는 말에 공감은 하지만 안바꼈으면 하는 바램이 있다..

 

Entity 복합 키 구현

컨설턴트님과 코치님의 피드백을 받은 후 DB 수정을 하던 도중 복합키 구현에 막혀있었다..

구글링을 하면서 @IdClass와 @Embeddable이라는 것을 알게되었고 안까먹기 위해 블로그 포스팅까지 마쳤다.

 

스켈레톤 코드 작성

개발을 시작하기 전에 스웨거로 URL을 알아보기 편하도록 Controller와 Dto부분을 작성하였다.

Controller는 메소드 선언만 해둔 상태라 개발 시작하면 작성할것이고, 설정부분은 처음부터 하지않을까 싶다.

 

기능 단위 분리

Jira 등록을 하기 위한 기능 단위 분리를 진행하였다.

 

포트폴리오 작성

Notion을 이용하여 나만의 포트폴리오를 작성중이다. (오늘 시작했고 적을게 있으면 계속 채워나갈 예정)

노션링크 ← 보신분은 피드백 부탁드립니다..

 

후기

생각보다 해야할게 많다.. 늦게나마 공부하기 잘했다는 생각이 든다.

팀장을 맡았지만 벌써부터 팀에 민폐가 되는것같아서 팀원분들께는 항상 죄송하다는 생각이 들고, 더 열심히 해야겠다는 생각밖에는 들지 않는다.

'잡담' 카테고리의 다른 글

8월 2주차 결산  (0) 2021.08.15
8월 1주차 결산  (0) 2021.08.08
solved.ac에 잔디가 생겼어요!  (0) 2021.07.27
스코페 2차 결과  (0) 2021.04.02
스코페 2021 후기  (0) 2021.03.28

복합 키를 구현하기 위해서 두 가지 방법을 이용할 수 있습니다.

  1. IdClass 어노테이션 이용
  2. Embeddable 어노테이션 이용

IdClass 어노테이션은 https://emoney96.tistory.com/249 에 정리가 되어있으며,

이 글에서는 Embeddable 어노테이션 적용'만' 다룰 것입니다.

 

우선 @Embeddable을 적용할 Id전용 클래스를 생성합니다.

@IdClass와 다른 점은 @Embeddable을 붙인다는 점밖에 없습니다.

FollowId 클래스를 생성합니다.

 

Follow Entity 클래스에서 EmbeddedId 어노테이션을 적용한 FollowId 클래스 변수를 선언합니다.

 

그 다음 @MapsId 어노테이션을 이용하여 FollowId의 각 변수들과 매핑시켜주시면 됩니다.

여기서 주의할 점은

User 클래스에서 작성한 mappedBy의 이름과 Follow의 변수명이 같아야하고,

Follow 클래스에서 작성한 MapsId의 value와 FollowId의 변수명이 같아야합니다.

 

FollowId의 변수 fan을 fan1으로 변경했을 경우에도 실행은 됩니다.

하지만 table은 의도한대로 만들어지지 않는 것을 알 수 있습니다.

위의 결과는 fan1은 외래 키가 아니라 follow table의 순수 컬럼이자 기본 키가 되는 것이고,

fan은 외래 키가 맞지만 FollowId 클래스에 매핑할 "fan" 변수가 없어서 기본 키가 되지 못한 상황입니다.

 

 

복합 키를 구현하기 위해 @Id 어노테이션을 두개 사용했더니 에러가 발생했습니다 ㅠ

 

구글링 하다가 @Id를 한 클래스에 두개를 쓰면 안된다는 것을 알게되었고 복합 키 구현 방법을 찾아다녔습니다.

 

일단 복합 키 구현에는 두 가지 방법이 있습니다.

  1. IdClass 어노테이션 이용
  2. Embeddable 어노테이션 이용

저는 이 글에서 IdClass 어노테이션 적용'만' 다룰 것입니다.

(Embeddable은 몰라서 그래요.. 공부해서 포스팅 해보겠습니다...)

 

 

우선 @IdClass를 적용하기 위해 Id전용 클래스를 생성합니다.

저는 FollowId 클래스를 생성하였습니다.

 

그 다음 Entity 클래스에서 IdClass 어노테이션을 추가해서 작성해둔 Id전용 클래스를 적어두면 끝입니다!

 

여기서 주의할 점은

User 클래스에서 작성한 mappedBy의 이름과 Follow의 변수명, FollowId의 변수명

이 세가지가 모두 같아야합니다!

'Spring' 카테고리의 다른 글

Entity @GeneratedValue with @IdClass, @EmbeddedId  (2) 2021.08.06
Entity @Embeddable을 이용한 복합 키 구현  (0) 2021.08.01
Spring @RequestParam String[] 문제  (0) 2021.07.24
Spring JPA Pagenation  (0) 2021.07.24
Spring Optional.isPresent()  (0) 2021.07.15

이전에 풀었던거 표시 안되는건 아쉽지만 다른 목표가 생겼습니다.. 헤헤

 

지금 웹 위주로 공부중이라 알고리즘을 안하고있지만..

10분만 투자해서 브론즈/실버 문제 풀면 가능하지 않을까?

'잡담' 카테고리의 다른 글

8월 2주차 결산  (0) 2021.08.15
8월 1주차 결산  (0) 2021.08.08
7월 5주차 결산  (2) 2021.08.01
스코페 2차 결과  (0) 2021.04.02
스코페 2021 후기  (0) 2021.03.28

+ Recent posts