데이터 접근 기술39 [Querydsl] 시작 초기에 jpql을 통해 동적쿼리나 Named Query를 작성하는 것을 벗어나서 이제 Querydsl을 사용하도록 변경하려고 한다. 정말로 동적 쿼리를 jpql을 사용하여 문자열로 합쳐서 사용하려니 중복코드와 휴먼에러가 정말 많이 발생한다. 또한 동적 쿼리에 대해 특정 조건이 추가되는 경우에 상당한 정신력을 소모하고 페이징 쿼리인 경우에 count를 조회하는 쿼리도 동일하게 중복해서 작성되기 때문에 매우 생산성이 떨어진다. Querydsl의 설정은 생각보다 복잡한 편이다. 하나씩 설정해 보자. https://start.spring.io/ Spring boot가 3.0 버전이 나왔다. 3.0 버전으로 진행하고 싶지만 3.0 버전의 default jdk 버전이 17이기에 일단 2.0 버전을 선택하여 만들려고.. 2023. 2. 1. DTO와 Entity DataTransferObject : 데이터 전송 객체 Entity : 도메인 여러분은 DTO를 어떻게 활용하고 계신가요? 저는 스프링 부트를 사용하여 개발의 생산성을 충분히 느끼면서 개발을 하고 있습니다. 이 중 API 통신에 있어서 가장 편리한 ArgumentResolver가 Jackson 라이브러리로 JSON 데이터를 Object에 매핑해주는 기능을 정말 잘 사용하고 있습니다. 우리가 사용하는 엔티티는 도메인의 한 부분으로 JPA를 사용하신다면 데이터베이스와 직접적으로 연결되어있습니다. 이 말은 데이터베이스가 변경되지 않는 이상 엔티티는 변경에 꽉 닫혀있어야 함을 말하고 있습니다. 그래서 우리는 외부에서 들어오는 여러 데이터를 엔티티로 직접 검증하고 받지 않습니다. 그 이유는 프레젠테이션 계층은 변.. 2022. 12. 11. 상속관계 매핑 관계형 데이터베이스에는 슈퍼 타입, 서브타입 관계라는 모델링 기법이 객체 상속과 유사하다. 따라서 상속관계 매핑이란 객체의 상속 구조와 DB의 슈퍼 타입 서브타입 관계를 매핑하는 것이다. 이러한 슈퍼 타입 서브타입의 논리 모델을 실제 물리 모델로 구현할 때 데이터베이스에서는 다음과 같은 방법을 제공한다. 1. 각각의 테이블로 변환한다. 이를 조인 전략이라고 한다. 2. 통합 테이블로 변환한다. 이를 단일 테이블 전략이라고 한다. 3. 서브타입 테이블로 변환한다. 구현 클래스마다 테이블 전략이라고 한다. JPA에서는 이러한 슈퍼 타입 서브타입 논리 모델을 여러 가지 방법으로 구현을 해도 전부 지원할 수 있다. 조인 전략 조인 전략은 정규화된 방식이다. 다음과 같이 매핑할 수 있다. @Entity @Inhe.. 2022. 12. 4. 값 타입 값 타입은 말 그대로 단순한 수치 정보이다. JPA에서 타입을 크게 분류하면 엔티티 타입과 값 타입이 존재한다. @Entity로 정의하는 객체는 엔티티 타입, 자바의 기본 타입이나 단순히 수치로 사용하는 타입을 값 타입이라 할 수 있다. 실제 우리가 값 타입을 활용할 수 있는 부분이 어디일까? 데이터베이스에서 여러 칼럼들을 모아 하나의 추상적인 객체로 표현할 수 있는 경우, 이를 애플리케이션 단에서 객체지향적으로 사용하고 싶다면 우리는 임베디드 타입을 고려해볼 수 있다. 복합 값타입이라고도하며 관련된 속성을 하나로 묶어 객체로 표현하는 것이다. 이는 또 다른 장점을 가지고 있는데, 클래스가 가지고있는 상태에 대해 책임을 질 수 있는 클래스가 존재하기 때문에 객체지향적으로 사용할 수 있다는 것이다. @Em.. 2022. 11. 20. OSIV open-session-in-view? OSIV는 기본값을 true로 설정되어있다. 이후 WARN 로그가 남는다. OSIV 전략은 트랜잭션 시작처럼 최초 데이터베이스 커넥션 시작 시점부터 API 응답이 끝날 때까지 영속성 콘텍스트와 데이터베이스 커넥션을 유지한다. 언제 JPA가 데이터베이스 커넥션을 가져올까? 트랜잭션 시작 시점에 영속성 콘텍스트가 데이터베이스 커넥션을 획득한다. 그럼 반환하는 곳은 어디일까? osiv가 켜져 있다면 트랜잭션이 끝나고 VIEW 계층까지 커넥션을 유지한다. 그렇기 때문에 우리가 view Template이나 API 컨트롤러에서 지연 로딩을 사용할 수 있던 것이다. 지연 로딩은 영속성 콘텍스트가 살아있어야 가능하고, 이는 기본적으로 데이터베이스와 커넥션을 유지한다. 다만 이 전.. 2022. 10. 31. JPQL Pagination 우선 페이징에 관한 여러 관련 지식들을 복습하자. 페이징을 하기 위해선 다음과 같은 정보가 필요하다. 1. 한 페이지에 출력될 데이터 수 2. 한 화면에 출력될 페이지 수 3. 현재 페이지 번호 우선 전체적인 페이지 수를 알아야 한다. 이를 위해 데이터를 몇 개씩 가져올지에 대한 변수를 정의한다. public int createTotalPage(int limit){ int totalData = getTotalData(); int totalPage = totalData / limit; float remainder = totalData % limit; if ( remainder > 0 ) { totalPage += 1; } return totalPage; } public int getTotalData(..... 2022. 10. 30. 벌크연산 보호되어 있는 글 입니다. 2022. 10. 29. N+1 문제 해결 JPA를 사용하기 위해 여러 공부를 하는 도중에 문제가 되는 상황을 발견했다. 도메인을 확인하고 테이블과 엔티티를 설계하는데 다대다 관계를 조인 테이블로 풀어 조인 테이블을 엔티티로 승격시켜 사용하는 것까지는 좋았다. 하나는 양방향 매핑이 필요해서 따로 작업을 해주고 나머지는 단방향 매핑으로 진행하고 나중에 필요하면 양방향으로 변경하자고 맘을 먹고 각종 테스트까지 마친 상태에서 테이블에 데이터를 넣고 객체 그래프 탐색을 통해 모든 데이터를 끌어올려 API로 전달하는 과정에서 문제가 발생했다. 간단히 문제가 발생한 부분을 설계하면서 느낀 점을 풀자 하나의 부분 모양이다. 현재 A, B에 관한 관계는 OneToOne이 현재 비즈니스 로직적으로 맞았지만 과거 기획에는 일대다 관계가 맞다고 생각했다. 그래서 우.. 2022. 10. 29. 나는 지금 JPQL이 필요하다! (2) 역시 줄임말을 펴줘야 한다 JPQL ( Java Persistence Query Language) JPA랑 앞글자가 거의 같다. 뭐 JPA가 자카르타로 변경됐다고 듣긴 했다 JPQL을 적용하기로 결정했는데 JPQL 문법을 조금 알아보자 엔티티와 속성은 대소문자를 구분한다. 다만 JPQL 키워드는 대소문자 구문 하지 않는다. 이 부분에서 매우 많은 오류가 예상된다. 별칭은 필수이다. as는 생략 가능하다. + 10/28 결국 동적 쿼리를 작성하다가 오류를 만났다. 띄어쓰기를 하지 않아 쿼리가 붙어버렸기 때문인데, 이에 대해서 인식하고 있어서 인지 오류를 스윽 보자마자 무슨 문제인지 확인했다. 만약 동적 쿼리가 많다면 jpql은 사용하지 않는 것이 좋다. 정말 코드도 길어지고 유지보수도 힘들다. 추가적으로 .. 2022. 10. 26. 나는 지금 JPQL이 필요하다! JPQL를 이해하지 않고 QueryDSL이 좋다고 라이브러리를 바로 쓸 수는 없다고 판단했다. 그래서 미래의 나에게 미리 사과를 하고 JPQL을 활용해 이번 프로젝트를 마무리할 것이다. 물론 요즘 DDD관련 서적도 읽고 JPA 공부도 다시 시작했고 JPQL도 배워 즉시 사용할 것이기 때문에 프로덕트 환경이 아닌 로컬에서 조금씩 리팩터링과 기술의 변경을 적용할 예정이다. JPA 및 Spring-Data-JPA가 아무리 좋다한들 복잡한 로직에 대해선 결국 쿼리 작성이 필요하긴 한 것 같다. 그래서 이를 지원하는 기능을 찾아보았다. JPQL JPA Criteria QueryDSL Native SQL SpringJdbcTemplate 아 순수 JDBC와 MyBatis는 제외하려고 한다. 현재 우리 프로젝트에 뭐.. 2022. 10. 26. 이전 1 2 3 4 다음