30%
61,600원
다른 수강생들이 자주 물어보는 질문이 궁금하신가요?
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
질문이 있습니다.
=========================================[질문 템플릿]1. 강의 내용과 관련된 질문인가요? (예)2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? (예)3. 질문 잘하기 메뉴얼을 읽어보셨나요? (예)[질문 내용]여기에 질문 내용을 남겨주세요. 안녕하세요 질문이 하나 있어서 글을 쓰게 되었습니다. 강의에서와 같이 조회용 샘플 데이터를 입력하고 h2 db에서 조회하였을 때 각 엔티티마다 pk가 연속적이지않고 아래 사진과 같이 설정되는데 혹시 이러한 이유가 있을까요? @GenerateValue 어노테이션과 관련이 있을까요?
- 해결됨실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
ordersV2 Result 객체로 감싸서 조회하는 코드
@GetMapping("/api/v2/simple-orders")public Result<SimpleOrderDTO> orders() { List<Order> orders = orderRepository.findByString(new OrderSearch()); List<SimpleOrderDTO> result = orders.stream() .map(SimpleOrderDTO::new) // 람다식 -> 메서드 참조 (Method Reference) .collect(Collectors.toList()); return new Result<SimpleOrderDTO>(result.size(), result);}@Data@AllArgsConstructorstatic class Result<T> { private int count; private List<T> simpleOrderDTO;} 이렇게 작성하면 될까요?? 결과 화면은 아래처럼 나옵니다!
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
OSIV 끈 상태에서
ordersV1을 실행하면 에러가 나는데요, ordersV1 위에 @Transactional 어노테이션을 붙이면 오류가 발생하지 않기는 하는데, 무언가 굉장히 이런 방식을 사용하면 안될 것 같은 느낌이 들어서요! 괜찮은 것일까요?
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
DTO 직접 조회 페이징
DTO를 직접 조회할때는 페이징을 할 수 있는 방법이 없는것인가요..???
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
JQPL 쿼리
이전 강의에 비슷한 질문을 남겼는데 주석을 처리한 코드를 실행해도 결과가 똑같습니다. 단지 주석을 처리한 코드를 실행하면 sql에서 크로스 조인이 사용되며, sql문장이 조금 달라지나, 똑같이 쿼리 한번이 나갑니다. 혹시 코드를 어떤 방식으로 작성하여도 상관없는 것인가요?
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
em.createQuery 에서 i.name과 oi.item.name 의 차이
학습하는 분들께 도움이 되고, 더 좋은 답변을 드릴 수 있도록 질문전에 다음을 꼭 확인해주세요.1. 강의 내용과 관련된 질문을 남겨주세요.2. 인프런의 질문 게시판과 자주 하는 질문(링크)을 먼저 확인해주세요.(자주 하는 질문 링크: https://docs.google.com/document/d/1j0jcJ9EoXMGzwAA2H0b9TOvRtpwlxI5Dtn3sRtuXQas/edit#heading=h.w2tomwsznga7)3. 질문 잘하기 메뉴얼(링크)을 먼저 읽어주세요.(질문 잘하기 메뉴얼 링크: https://docs.google.com/document/d/1xCQKit-1V6l6ObeCe49St33RHPzLF_P_c3o7aSDTKs0/edit#heading=h.7dhnp46ven0v)질문 시에는 위 내용은 삭제하고 다음 내용을 남겨주세요.=========================================[질문 템플릿]1. 강의 내용과 관련된 질문인가요? (예/아니오)2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? (예/아니오)3. 질문 잘하기 메뉴얼을 읽어보셨나요? (예/아니오)[질문 내용]선생님이 짜주신 코드에서, 궁금증을 느껴 i.name을 oi.item.name으로 바꾸고 join oi.item i 부분을 지웠습니다만, 출력에는 전혀 지장이 없었는데 inner join 대신 join만 생겼습니다. 쿼리가 나가는 횟수도 동일한 것 같은데 혹시 i.name으로 굳이 해주신 이유가 있으신가요? oi.item.name으로 사용해도 될까요? 또한 같은 문맥으로 where oi.order.id in :orderIds부분을 " join oi.order oio where oi.order.id in :orderIds"이렇게 고쳐도 join문이 늘어날 뿐 문제가 없었습니다. 아무튼 i.name과 oi.item.name의 차이에 대해서 궁금합니다. 굳이 별 차이 없으면 item join문을 작성해주지 않는 것이 더 코드가 깔끔해 보여서요
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
spring MVC와 REST API차이의 개념이 궁금합니다.
제가 이해한건 Spring MVC와 REST API 개발의 차이는 VIEW를 반환하는가 아니면 JSON의 데이터를 반환하는가의 차이인데 제가 이해한것이 맞나요??
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
안녕하세요 강사님 JPA join에 관련해서 질문이 있습니다.
안녕하세요. 질문은 2가지 정도가 있는데 다음과 같습니다 1. DTO를 반환하는 형태의 JPQL를 작성할 때 join문을 사용해서 쿼리를 작성해주셨습니다. 문득 로그를 보다가 든 생각이 DTO가 아니라 entity를 조회하기 위해 JPQL을 작성할 때, join을 사용하면 즉시로딩이나 지연로딩 관계없이 연관객체에 대해서 N+1문제가 발생하는 걸로 알고 있습니다. 근데 DTO를 반환하는 형태로 JPQL를 작성할 때는 join문을 사용해도 쿼리 한번에 연관객체들의 정보를 가져오더라구요 (강의에서 Order의 Member나 Delivery) 이러한 차이의 이유가 JPQL이 실행 될 때 JPA에서 관리를 하냐 안하냐의 차이로 볼 수 있을까요? 2. JPA는 트랙잭션 안에서만 동작한다고 알고 있습니다. 그런데 강의를 보다보면 Controller에서 Repository를 바로 의존성 주입받아서 사용하시더라구요. @Transaction은 서비스에만 붙어있고 Repository에는 선언되어 있지 않은데 정상적으로 동작하는 것은 조회는 트랙잭션 관계없이 JPA에서 사용이 가능한것인가요? 그 서비스 코드를 작성할 때 조회 메서드는 readOnly속성을 true로 주면서 작성했던 걸 보면 조회할 때도 @Transaction이 선언되어야 동작하는것이 아닌가 하는 생각이 들어서 질문 남겨드립니다. 항상 명강의 해주셔서 감사합니다. ps. MSA와 관련한 강의는 계획이 없으신가요? ㅎㅎ
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
컬렉션의 값이 출력되는 이유가 뭘까요?
안녕하세요 강의를 듣던 와중에 궁금한 점이 생겨서 질문드립니다. @GetMapping("api/v1/orders") .... orderItems.stream().forEach(o->.getItem().getName());과 같이 출력했을 때 value는 name만을 출력했는데 postman으로 결과값 조회시 어떻게 OrderItems의 모든 값들이 출력되는 건가요?
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
스트림으로 DTO만들거나 for문으로 DTO로 변환 질문드립니다.
강사님 안녕하세요 1편에 이어 2편도 수업을 정말 잘듣고 있습니다. 다름이 아니라 회원 조회시 v1으로 했을 땐 Entity를 그대로 노출 하거나 하면 위험하기도 해서 v2에선 DTO로 변환을 해서 Result 라는 객체에 DTO를 보내셨는데요. 정말 궁금한건 데이터가 실무에선 1~10건 이정도가 아니고 1만건을 줘야 한다거나 이런 경우도 충분히 발생할 수 있다고 보는데 그렇게 되면 강사님이 알려주신데로 for문 또는 stream으로 변환을 해서 보내게 되면 클라이언트가 응답할 시간이 좀 많이 늘어날거라고 생각되는데 거기에 대해서는 따로 속도 이슈라던지 성능 이슈라던지 생겼을것 같습니다. 제 짧은 생각으로는요! 그런경우는 어떤식으로 처리를 하셨는지.. 정말 비법이 궁금합니다. 다른 영상 찾아보고 라이브러리 찾아봐도 바로 변경 해주는 것도 있긴한데 솔직히 직관적으로 개발 하려면 강사님이 알려주신 방법이 좋다고 보는데요.. 이런 이슈가 발생한적이 혹시 없으신지요...
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
v1: 엔티티직접노출 방식에 달린 주석 질문
@GetMapping("/api/v1/orders")public List<Order> ordersV1() { List<Order> all = orderRepository.findAll(); for (Order order : all) { order.getMember().getName(); //Lazy 강제 초기화 order.getDelivery().getAddress(); //Lazy 강제 초기환 List<OrderItem> orderItems = order.getOrderItems(); orderItems.stream().forEach(o -> o.getItem().getName()); //Lazy 강제 초기화 } return all;} 첨부된 소스코드를 보면 "트랜잭션 안에서 지연 로딩 필요"라고 v1 메소드에 설명이 있습니다. orderRepository.findAll( ); 의 호출이후 트랜잭션은 종료됐을 텐데 그 이후에 강제 Lazy로딩을 하는 것이 위 주석의 설명과 혼동이 됩니다. 기본편에서 진행을 할 때는 항상 tx.commit( ) 이전에 Lazy로딩을 했기 때문에 트랜잭션 안에서 지연로딩을 한다는 의미를 당연하게 받아드렸지만 웹MVC를 결합하면서 Controller쪽에서 findAll() 호출 후 진행되는 상황은 트랜잭션이 종료된 후이기 때문입니다. 답변부탁드립니다. 감사합니다.
- 해결됨실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
return new CreateMemberResponse(id); 필요성
@RestController @RequiredArgsConstructor public class MemberApiController { .. @PostMapping("/api/v1/members") public CreateMemberResponse saveMemberV1(@RequestBody @Valid Member member) { Long id = memberService.join(member); return new CreateMemberResponse(id); } @Data static class CreateMemberResponse { private Long id; public CreateMemberResponse(Long id) { this.id = id; } } 일 때, memberService.join(member); 의 반환 값이 엔티티가 아니고 단순히 Long id인 값인데도 return id; 로 안하고 CreateMemberResponse 클래스를 만들어 return new CreateMemberResponse(id); 로 하는 이유는 뭔가요 ? 처음에는 Long id = memberService.join(member); return id; 로 했다가 누군가가 inline으로 합쳐버려서 return memberService.join(member); 으로 만들어 버리면 memberService.join의 반환 값이 수정될 때 컴파일이 안뜨고, api에서 스펙이 바뀌어 오류가 생길 가능성이 있어서 컴파일 단계에서 막아버릴려고 CreateMemberResponse 클래스를 만들어 return new CreateMemberResponse(id); 를 해주는 것일까 생각도 들었는데, inline으로 합쳐버리고 memberService.join의 반환값이 바뀌어도 어차피 @PostMapping("/api/v1/members") public Long saveMemberV1(@RequestBody @Valid Member member) { Long id = memberService.join(member); return id; } 인 상태에서 inline하는 거니까 public Long saveMemberV1(..) {..} 에서 return값이 Long타입이 아니게 바뀌면 컴파일 뜰 것 같아서 그것도 아닌가 싶기도 하고 .. 유지보수할 때, 추적하기 쉬우려고 그러는 건가 싶기도 하고 .. 갑자기 든 생각인데, 일반적으로는 단순히 id만 반환할 일이 없으니 일반적인 케이스를 생각해 만드신 건가 싶기도 하고.. 이 케이스만 예외적으로 Long으로 써도 가능한 건지 궁금합니다 :]
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
변경감지 엔티티 set
엔티티를 변경감지해서 update하는 경우 setter를 사용하시는데 엔티티에 @setter를 사용하되 엔티티내에서 따로 수정하는 필드를 변경하는 메서드를 만들어야 하는것이 유지보수때 더 나을까요? 엔티티에는 getter만 사용하신다고 하셨는데 변경 감지때는 setter를 꼭 사용해야되는 것 같아서 질문합니다.
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
종속성 순환에러 질문드립니다....
선생님 안녕하세요! 꾸준히 걸어가고있는 코린입니다.. 혹시 @Component@Transactional@RequiredArgsConstructorstatic class InitService { 이 부분에서 static이 아닌 private으로 봐꿔보면 종속성 순환에러에 걸리는데 static일때는 메모리를 공유하고 private일때는 공유 하지않아서 생기는 문제인가요?
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
안녕하세요 강사님! 회원등록 API 관련 400에러 문의 드립니다
안녕하세요 강사님! 좋은 강의 잘 듣고 있습니다:) 다름이 아니라 해당 강의 회원등록 API 파트를 수강 중 포스트 맨에 json을 넘길 때 400에러가 발생하여 문의 드립니다. 이전 5월에 다른 분께서 질문 주신 게시판 글을 찾았는데 해결을 어떻게 하셨는지 알 수가 없어서요. api 컨트롤러 코드 확인을 했는데도 어느 부분을 고쳐야 하는지 모르겠어서 문의 남깁니다. 그럼 좋은 저녁 보내세요!
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
OneToMany 데이터 조회 시
안녕하세요! Order와 OrderItem을 가져올 때 N+1, 페이징 처리와 더불어 OrderItem 데이터 컬럼을 기준으로 동적 정렬조건이 필요하다면 어떤식으로 처리해야 할까요?
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
QClass가 안보입니다.
현재 gradle을 빌드해서 Q 클래스들이 생성이 됬는데 안보여서 들어가보니 명시된 package 경로로 보아도 안보입니다.. 이게 왜이럴까요? 개발엔 큰 문제가 없을까요? 현재 gradle 소스 남깁니다.. buildscript { dependencies { classpath("gradle.plugin.com.ewerk.gradle.plugins:querydsl-plugin:1.0.10") }}plugins { id 'org.springframework.boot' version '2.5.4' id 'io.spring.dependency-management' version '1.0.11.RELEASE' id 'java'}group = 'jpabook'version = '0.0.1-SNAPSHOT'sourceCompatibility = '11'apply plugin: 'io.spring.dependency-management'apply plugin: "com.ewerk.gradle.plugins.querydsl"repositories { mavenCentral()}dependencies { implementation 'org.springframework.boot:spring-boot-starter-data-jpa' implementation 'org.springframework.boot:spring-boot-starter-thymeleaf' implementation 'org.springframework.boot:spring-boot-starter-validation' implementation 'org.springframework.boot:spring-boot-starter-web' implementation 'com.github.gavlyukovskiy:p6spy-spring-boot-starter:1.5.6' implementation 'com.fasterxml.jackson.datatype:jackson-datatype-hibernate5' compileOnly 'org.projectlombok:lombok' developmentOnly 'org.springframework.boot:spring-boot-devtools' runtimeOnly 'com.h2database:h2' annotationProcessor 'org.projectlombok:lombok' testImplementation 'org.springframework.boot:spring-boot-starter-test' testImplementation("org.junit.vintage:junit-vintage-engine") { exclude group: "org.hamcrest", module: "hamcrest-core" } implementation 'com.querydsl:querydsl-jpa' implementation 'com.querydsl:querydsl-apt'}def querydslDir = "$buildDir/generated/querydsl"querydsl { library = "com.querydsl:querydsl-apt" jpa = true querydslSourcesDir = querydslDir}sourceSets { main { java { srcDirs = ['src/main/java', querydslDir] } }}compileQuerydsl{ options.annotationProcessorPath = configurations.querydsl}configurations { querydsl.extendsFrom compileClasspath}test { useJUnitPlatform()} 일단 ppt 코드로 넣었구요.. 원래는 'src/main/java/generated/querydsl' 의 하위에 만들어져야 하는게 맞나요?
- 해결됨실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
queryDSL Gradle 설정
다른 질문글을 보고 gradle 내용을 비교해봐도 차이를 모르겠습니다.. https://www.inflearn.com/questions/149157 이 분과 같은 상태인데 답변 내용을 봐도 해결을 못하겠습니다 ㅠ buildscript { dependencies { classpath("gradle.plugin.com.ewerk.gradle.plugins:querydsl- plugin:1.0.10") }}plugins { id 'org.springframework.boot' version '2.5.4' id 'io.spring.dependency-management' version '1.0.11.RELEASE' id 'java'}group = 'jpabook'version = '0.0.1-SNAPSHOT'sourceCompatibility = '11'apply plugin: "com.ewerk.gradle.plugins.querydsl"configurations { compileOnly { extendsFrom annotationProcessor }}repositories { mavenCentral()}dependencies { implementation 'org.springframework.boot:spring-boot-starter-data-jpa' implementation 'org.springframework.boot:spring-boot-starter-thymeleaf' implementation 'org.springframework.boot:spring-boot-starter-validation' implementation 'org.springframework.boot:spring-boot-starter-web' implementation 'com.github.gavlyukovskiy:p6spy-spring-boot-starter:1.5.6' implementation 'com.fasterxml.jackson.datatype:jackson-datatype-hibernate5' compileOnly 'org.projectlombok:lombok' developmentOnly 'org.springframework.boot:spring-boot-devtools' runtimeOnly 'com.h2database:h2' annotationProcessor 'org.projectlombok:lombok' testImplementation 'org.springframework.boot:spring-boot-starter-test' testImplementation("org.junit.vintage:junit-vintage-engine") { exclude group: "org.hamcrest", module: "hamcrest-core" } implementation 'com.querydsl:querydsl-jpa' implementation 'com.querydsl:querydsl-apt'}def querydslDir = "$buildDir/generated/querydsl"querydsl { library = "com.querydsl:querydsl-apt" jpa = true querydslSourcesDir = querydslDir}sourceSets { main { java { srcDirs = ['src/main/java', querydslDir] } }}configurations { querydsl.extendsFrom compileClasspath}compileQuerydsl{ options.annotationProcessorPath = configurations.querydsl}test { useJUnitPlatform()}
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
커맨드와 쿼리
학습하는 분들께 도움이 되고, 더 좋은 답변을 드릴 수 있도록 질문전에 다음을 꼭 확인해주세요.1. 강의 내용과 관련된 질문을 남겨주세요.2. 인프런의 질문 게시판과 자주 하는 질문(링크)을 먼저 확인해주세요.(자주 하는 질문 링크: https://docs.google.com/document/d/1j0jcJ9EoXMGzwAA2H0b9TOvRtpwlxI5Dtn3sRtuXQas/edit#heading=h.w2tomwsznga7)3. 질문 잘하기 메뉴얼(링크)을 먼저 읽어주세요.(질문 잘하기 메뉴얼 링크: https://docs.google.com/document/d/1xCQKit-1V6l6ObeCe49St33RHPzLF_P_c3o7aSDTKs0/edit#heading=h.7dhnp46ven0v)질문 시에는 위 내용은 삭제하고 다음 내용을 남겨주세요.=========================================[질문 템플릿]1. 강의 내용과 관련된 질문인가요? (예/아니오)2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? (예/아니오)3. 질문 잘하기 메뉴얼을 읽어보셨나요? (예/아니오)[질문 내용]여기에 질문 내용을 남겨주세요. 강사님께서 커맨드와 쿼리를 분리하는 방식을 자주 언급하시는데, 정확히 커맨드와 쿼리가 어떤 차이가 있나요??
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
DTO질문
안녕하세요! DTO 생성자 관련해서 궁금증이 생겨 글 남깁니다. 아래 캡쳐처럼 DTO를 inner클래스가 아닌 그냥 클래스로 만든 후 여러 곳에서 사용 중인데 DTO에서 각각의 API에 맞게 생성자를 여려개 만들어서 다른곳에서 각각 다른 생성자를 통해 Result를 생성해서 사용해도 문제가 없는지 궁금합니다! @Datapublic class GymDto { private String gymName; private List<MemberDto> memberList; public GymDto(Gym gym) { memberList = gym.getMemberList().stream().map(m -> new MemberDto(m)).collect(Collectors.toList()); } public GymDto(String gymName) { this.gymName = gymName; }}