월 17,600원
5개월 할부 시다른 수강생들이 자주 물어보는 질문이 궁금하신가요?
- 미해결실전! 스프링 부트와 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; }}
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
MVC, API 중 어느것을..
[질문 템플릿] 1. 강의 내용과 관련된 질문인가요? (예/아니오) 예2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? (예/아니오) 예3. 질문 잘하기 메뉴얼을 읽어보셨나요? (예/아니오) 예[질문 내용] 안녕하세요. 좋은 강의 만들어주셔서 감사합니다. 강의를 다보고 궁금한 것이 생겼습니다. Form과 MVC로도 애플리케이션을 만드는 것이 가능한데 언제 API를 사용하고 언제 MVC를 사용해야 하는지 궁금하네요. 예를 들어, 주문정보를 수정하는 페이지에서 Post처리를 MVC로도 가능하고 API로도 가능한데.. 확장성을 위해서 데이터를 조작하는 기능은 전부 API로 만드는 것이 맞나요? 혼자 고민을 해본 결과 뷰를 보여주는 것만 MVC를 하고 나머지는 API로 만드는게 맞는거 같다라는 생각인데 영한님 한테 조언을 듣고 싶습니다. ps. 스프링 시큐리티관련 강의는 예정이 없으신가요?ㅎㅎ..
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
RestApi에 대하여
안녕하세요! 강사님 항상 좋은강의 정말 감사드립니다. 평소에 Api 특히 Rest APi에 대해서 너무 광범위한 사용과 문맥에 정말 개념이 잡기가 어려웠는데, 이번 강의를 통해서 많이 개념이 잡혀나가는것 같아 뿌듯하고 감사합니다. 공부를 하던중에 궁금한 것이 있는데 1. Rest api라고 하는건 mvc패턴에서의 api(템플릿 엔진을 주로 사용하여 리턴해주는 컨트롤러)는 Rest api의 범주에 들어오지 않는건가요? 2. 그리고 클래스 Annotation중에 @RestController가 붙으면 http body 부분에 json 형식으로 data가 디폴트로 나가는 건지 궁금합니다. 데이터를 text/plain이나, xml같은 형식으로 body에 response하려면 어떻게 해야할까? 라는 궁금증이 있습니다. 답변 부탁드립니다!
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
@Scheduled 사용 시 @Transactional 메소드 사용 실패 관련
안녕하세요, 강의 잘 듣고 있습니다!! JPA 사용 중에 강의만으로는 어려운 문제를 만나 문의드립니다! JPA 트랜잭션에 대한 오류인것 같고 꽤 오래 찾아았으나, 아직 답을 찾지 못했습니다. 현상은 @Scheduled 가 추가된 메소드에서 다른 서비스 A의 @Transactional 메소드를 호출 시 아래와 같은 에러가 발생합니다. 예시) @Transactional public void changeStatus(Long id, String status) { Obj o = objRepository.findById(id).get(); o.setStatus(status); } 환경 SpringBoot 2.4 이며, 자동으로 설정되는 JpaTransactionManager 를 사용합니다. ============== 에러 메시지 ============== 2021-08-30 18:14:17.536 ERROR 18252 --- [ scheduler-1] o.s.s.s.TaskUtils$LoggingErrorHandler.handleError:95 - Unexpected error occurred in scheduled task org.springframework.orm.jpa.JpaSystemException: Unable to perform beforeTransactionCompletion callback: No thread-bound request found: Are you referring to request attributes outside of an actual web request, or processing a request outside of the originally receiving thread? If you are actually operating within a web request and still receive this message, your code is probably running outside of DispatcherServlet: In this case, use RequestContextListener or RequestContextFilter to expose the current request.; nested exception is org.hibernate.HibernateException: Unable to perform beforeTransactionCompletion callback: No thread-bound request found: Are you referring to request attributes outside of an actual web request, or processing a request outside of the originally receiving thread? If you are actually operating within a web request and still receive this message, your code is probably running outside of DispatcherServlet: In this case, use RequestContextListener or RequestContextFilter to expose the current request. at org.springframework.orm.jpa.vendor.HibernateJpaDialect.convertHibernateAccessException(HibernateJpaDialect.java:331) Trace 로그로 볼 때에는 transaction marked for rollback only 라는 메시지도 나옵니다. 변경 감지로 할 경우와, save 메소드를 호출하는 경우 모두 발생하구요.. 제가 무언가 놓치고 있는 게 있다면, 조언 부탁드립니다!!
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
spring boot 내장 톰캣 관련 질문
안녕하세요 선생님 컨트롤러에서 쿠키를 세팅하기 위해서 구글링 중에 컨트롤러에서 쿠키를 세팅 하기 위해서는 서블릿 클래스를 사용 해야 한다는 것을 알았습니다 그런데 제가 알기로는 스프링부트를 사용하지 않고서 아파치 톰캣을 사용할 때 사용자 설정 서블릿 클래스를 만들고 (java에서 제공하는 서블릿 상속받아서) 아파치 톰캣에서 제공하는 컴파일러로 빌드 후 나온 .class 파일을 톰캣폴더에 넣어서 url 매핑을 통해서 사용자에게 response를 보내는 것으로 알고 있습니다. 그런데 이 과정에서 궁금점이 발생 하게 되었습니다. 톰캣이 해당 url 에 http 패킷을 분석해서 httprequestbody와 httpresponsebdy를 만드는 것 까지는 알겠는데 별도의 서블릿 클래스 설정없이 어떻게 컨트롤러 parameter로 httpservletrequest와 response가 넘어가는지 궁금합니다
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
Service 의 반환값이 DTO인지 entity 인지에 대해서 궁금한점이 있습니다.
안녕하세요, 영한님 . 쓰신 책으로 먼저 이름을 접하게 되었는데 인프런에도 강의하고계셔서 신기합니다 ㅎㅎ 다름이 아니고 작은 프로젝트를 하는데 궁금한점이 생겨서 질문드립니다. 제가 아는 바로는 Service의 반환값은 entity가 아닌 DTO로 알고있습니다. https://stackoverflow.com/questions/21554977/should-services-always-return-dtos-or-can-they-also-return-domain-models 그런데 궁금한점은 만약 Service A의 특정메소드의 반환값을 Service B에서 필요로 하게 될때 어떻게 처리할지 고민입니다. 만약 ServiceA의 메소드의 반환값이 entity면 Service B 에서 큰 작업 없이 사용하면 되는데, dto인 경우일때 처리가 쉽지않은것같습니다. 그래서 ServiceA가 사용하는 repository A를 Service B에서 호출하는 방식을 사용을 고민중인데 혹시 이게 나쁜 방법일지 잘 모르겠습니다. 혹시 DTO들을 entity로 변환하는 로직을 모두 넣어줘야할까요? 정리하자면 Service A,B의 메소드가 entity가 아닌 DTO를 반환한다. ServiceB에서 A entity에 대해서 접근할때 Service A는 dto를 반환해서 B에서 원할히 접근하기 어렵다. ServiceB에서 ServiceA를 호출하지 않고 repositoryA를 호출하는데 좋은 방식인지 모르겠다. 조언 부탁드립니다 !
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
DTO 질문 드립니다
안녕하세요. 현재 영한님 로드맵 따라 JPA활용2편까지 수강 후 혼자 토이프로젝트를 진행 중 입니다. 프로젝트를 진행하다보니 Item과 Book처럼 추상클래스를 상속받는 경우에 DTO를 어떻게 만들면 좋을지 고민하다 해답이 안나와서 질문드립니다. 예를 들어 ItemRepository에서 findAll() 메서드를 호출하여 List에 전체 Item목록을 담고 있고 List에 book만 있는게 아니라 book, movie 등등 여러가지 타입이 있는경우 Dto를 어떻게 설계하는게 좋은 방법인가요?
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
다대다 cascade remove 관련 질문이있습니다.
안녕하세요 선생님 수업을 듣고 토이 프로젝트 진행중 막히는 부분이 있어 질문드립니다.다대다 관계를 일대다 다대일로 풀어서 총 3개의 엔티티에서 양방향 연관관계를 가져가는 상황입니다.이때 서비스계층에서 '다'에 해당하는 가운데 엔티티를 이용해 3개의 연관관계를 한번에 끊고자 cascade.ALL을 사용했는데오류가 발생했습니다.저는 이게 1 -> 2 <- 3 이런식의 일 다 일 관계라면 2에서 엔티티를 삭제할때 1,3의 엔티티가 같이 삭제되는데 그러면 1,3과 현재 연관관계를맺고 있는 다른 DB레코드에도 영향을 주기때문에 ConstraintViolationException 예외가 발생한다고 생각했습니다.결국 선택한 방식은 3개의 repository에서 각각 지워주는 방식으로 진행했는데 이러면 코드 라인이 3줄이 되고 쿼리도 3번이 나가서성능적인 이슈가 있다고 생각하는데 다대다를 일대다 다대일로 풀어낸 상황에서 cascade를 사용해 한 코드 라인으로 delete해줄수 있는 방법이 없을까요??
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
CSR시 질문드립니다
강의 내용 자체가 SSR 기준일 경우의 설명인 것 같습니다. 그렇다면 CSR 일 경우에는 OSIV를 켜놓으면 SSR 보다 커넥션을 더 짧게 가지고 있을 것 같은데요 ! Vue.js같은 프론트 프레임 워크를 사용하고, 컨트롤러에서 @RestController를 사용해 데이터만 ResponseEntity에 담아 내려줄 경우 OSIV로 인한 리스크가 크게 줄어드는 걸까여?
- 미해결실전! 스프링 부트와 JPA 활용2 - API 개발과 성능 최적화
n+1
안녕하세요 영한님 20:07 부분에서 n+1 현상이 나타나신다고 말씀을 주셨는데 이 부분은 루프를 돌면서 컬렉션을 직접 추가한 부분이라 어쩔수 없는 상황아닌가요?? 결과론적으론 N+1 현상이 나타났다고 이해 하면 될까요?