묻고 답해요
164만명의 커뮤니티!! 함께 토론해봐요.
인프런 TOP Writers
-
해결됨실전! 스프링 부트와 JPA 활용1 - 웹 애플리케이션 개발
동일한 레코드의 참조 값을 가지고 있는 영속성 컨텍스트의 동일 객체를, 동시에 서로 다른 값으로 업데이트하여 커밋할 때의 작동 방식
안녕하세요. 추천해주신 야생형 스타일에 따라 활용 1편부터 듣고 있는 수강 중인 학생입니다. 매번 강의 너무 잘 듣고 있습니다. 제가 스프링에 대한 이해도가 아직 낮기도 하고, 다른 강의를 이어서 듣다보면 해결될 문제일진 모르겠지만, 이해가 잘 안되는 부분이 자꾸 생각나서 질문 남깁니다. @GetMapping("/test1") @ResponseBody public String test1() { log.info("test1 controller"); try { Thread.sleep(5000); } catch(Exception e) { } log.info("test1 controller finished"); return "test1 finished"; } @GetMapping("/test2") @ResponseBody public String test2() { log.info("test2 controller"); try { Thread.sleep(5000); } catch(Exception e) { } log.info("test2 controller finished"); return "test2 finished"; } 위와 같은 컨트롤러 메서드 코드가 있다고 가정할 때, 각 컨트롤러 메서드가 다른 메서드라면, 각각의 메서드 별로 스레드를 가진 채 실행한다고 이해가 되었습니다. 즉, test1이 실행하는 도중에 test2가 실행될 수는 있지만, test1이 실행하는 도중에 test1은 중복해서 실행될 수 없는 것처럼 보였습니다. 일단 스프링의 작동 방식은 이처럼 이해되었습니다. 그런데 만약 아래와 같이 test1과 test2에서, 동일한 Item객체를 JPA에서 동시에 꺼내온 상황에 업데이트가 일어나면 어떻게 되는지 궁금합니다. 아래의 실행 과정은 제가 개인적으로 생각해본 과정인데, 어떠한 부분이 잘못되었는지 지적해주시면 감사하겠습니다. 1. test1에서 서비스 로직을 실행하는 도중 JPA를 이용하여 Item 객체를 findOne해서 꺼내온다(동시에 영속성 컨텍스트에 등록이 된다는 것처럼 이해되었습니다.) 2. test1가 아직 실행되고 있는 와중에, test2에서 서비스 로직을 실행하는 도중 JPA를 이용하여 Item객체를 findOne해서 꺼내온다(이 객체 또한 영속성 컨텍스트에 등록이 된다는 것처럼 이해되었습니다.) 3. 현재 test1과 test2에서 각각 동일한 Item 객체를 가져와서 영속성 컨텍스트에 등록이 된 상황이라고 보겠습니다. 앞으로 실행 될 test1의 서비스에서는 count를 10올릴 것이고, test2에서는 count를 10내린다고 가정해보겠습니다. 현재 Item 객체에는 100이라는 값이 저장되어있습니다. 4. test1에서 item.addCount(10)을 하면, count는 110이 될 것입니다. 바로 이어서 test2에서 item.removeCount(10)을 하면, count는 90이 될 것입니다. 5. test1의 서비스 로직이 끝나면, @Transactional 어노테이션을 통해 commit이 일어나고, dirty checking을 하며 item 객체의 count를 110으로 업데이트하는 쿼리문을 날릴 것입니다. 6. test2의 서비스 로직이 끝나면, @Transactional 어노테이션을 통해 commit이 일어나고, dirty checking을 하며 item 객체의 count를 90으로 업데이트하는 쿼리문을 날릴 것입니다. 7. 따라서 최종적으로 item 객체의 count값은 90으로 업데이트 될 것입니다. 하지만 실제로는 test1에서 10을 더하고, test2에서 10을 차감하였으니, 동일한 item 객체에 대한 count값은 DBMS 상에서 100으로 유지되어야 맞을 것입니다. 제가 생각한 실행 과정은 90으로 값이 업데이트되며 DBMS의 값의 일관성을 깨뜨리는 상황입니다(물론 제가 짧은 생각대로 실행한 과정의 결과가 90이란 것이지, 코드의 실행 결과가 90이라고 단언한 것은 아닙니다. 결과도 궁금하지만, 100이라는 결과가 나오는 과정에서 어떻게 실행되는지가 궁금한 것입니다!). 어떠한 부분이 잘못되었고, 그 부분은 어떻게 해결되어지는 것인지 궁금합니다. 개인적으로 생각해본 가정은 다음과 같습니다. 가정1 : 동일한 레코드를 조회한 객체에 대해서는 영속성 컨텍스트에 동일한 객체로 기억되기때문에 test1과 test2에서는 동일한 참조 값을 가진 item 객체를 가지고 있다. 따라서 addCount를 할 때 110으로 바뀌고, removeCount를 할 때 100으로 다시 바뀌기 때문에, test1과 test2의 커밋 각각에서는 count 값을 100으로 바꾸는 동일한 update문이 두 번 일어난다. 가정 2 : 동일한 레코드에 대해서 이미 commit 또는 업데이트 된 내역이 있으면, 지금 일어나는 commit은 그냥 ROLLBACK을 시켜버린다. 하지만 이렇게 할 경우, 자바 코드 상에서 DBMS에 저장된 아이템에 대해 동일한 것을 접근했는지 어떻게 기억할 것이며, rollback으로 인한 오버헤드는 감수하는 것인지 의문점이 남습니다. 가정 3 : 동일한 레코드로 조회된 객체에 대해서는 업데이트가 일어나는 전 과정에, 해당 객체에 lock을 걸어둔다. 그러면, 업데이트가 끝나서 commit이 되고, lock을 해제할 때까지 해당 객체에는 접근하지 못한다. 처음에는 이러한 과정이 @Transactional 어노테이션이 붙어있으면, 이 어노테이션이 달려있는 메서드 중에 1개씩만 실행되면서 수행되는 줄 알았는데 제가 아직 개념이 부족한 탓인지 딱히 그렇게 실행되는 것 같지는 않았습니다. 가정 4 : 애초에 이러한 설계가 잘못된 것이다. test1과 test2에서 동시에 item 객체를 수정하는 과정의 코드는 없어야한다. test1 또는 test2, 둘 중에 하나의 메서드에서만 item 객체를 수정할 수 있어야한다. 가정 5 : 그냥 DBMS에서의 트랜잭션처럼 관리된다. 자바 상에서 동시에 실행되는것처럼 보여도 JPA를 통해 serializable한 실행 결과를 보장해준다. 일단 DBMS에서 트랜잭션 간에 동시성을 관리하는 체계를 생각하면, 위와 같은 가정들이 나온다고 생각했습니다. 하지만 그건 DBMS에서의 동시성 관리 체계이지, JPA 상에서도 @Transactional 어노테이션 하나로 그것처럼 동일하게 관리되는지는 잘 모르겠어서 의문이 남습니다. 질문이 미흡해서 제가 의문점을 제대로 남긴 것인지 모르겠네요. 바쁘신 와중에 시간 내 주셔서 감사합니다. +++ @Test@Transactional@Rollback(false)public void 동시업데이트() { // given Book book = new Book(); book.setName("희재의 책"); book.setIsbn("1234"); book.setStockQuantity(100); book.setPrice(30000); em.persist(book); Book book1 = em.find(Book.class, 1L); Book book2 = em.find(Book.class, 1L); // when book1.addStock(10); book2.removeStock(20); Book book3 = em.find(Book.class, 1L); // then assertThat(book1).isSameAs(book2); assertThat(book2).isSameAs(book3); assertThat(book3.getStockQuantity()).isEqualTo(90);} 일단 위와 같은 테스트 코드로 DBMS 상에서 같은 레코드를 조회한 아이템에 대해서는 동일한 객체를 반환하는 것을 확인했습니다. (같은 트랜잭션 상에서만?) 그런데 처음 적은 예시처럼 영속성 컨텍스트에 등록된 동일한 객체에 대해 동시에 커밋이 여러 개 일어나면 어떻게 되는지, 또 이것을 확인해보고 검증하는 테스트 코드는 어떤 식으로 작성해야하는지 잘 모르겠네요.. ㅠㅠ
-
미해결[리뉴얼] React로 NodeBird SNS 만들기
질문있습니다
제로초님 5강의 프로젝트를 실행 시키고나서 로그인 로그아웃 글쓰기, 팔로워등 하면 잘되다가 이런 오류가 뜹니다. 이런 오류는 뜨는이유가 무엇이며, 어떻게 해결해야할지 궁금합니다ㅎㅎ 질문 2 프론트랑 백앤드랑 cmd창에 어떤 순서로 npm run dev를 하는게 좋나요?
-
미해결파이썬 알고리즘 문제풀이 입문(코딩테스트 대비)
3중 for문 대신에 combinations 써도 되지 않나요?
저는 combinations 쓰고 sum() not in list일때 sum으로 묶어서 해줬었는데.. 맞을까요?
-
해결됨프론트엔드 개발환경의 이해와 실습 (webpack, babel, eslint..)
"웹팩 기본편 - 자주 사용하는 플러그인" 강의내용중 질문이 있습니다.
안녕하세요 웹팩을 webpack.base.babel.js로 따로 관리하는데 2가지 질문이 있습니다. 1. ESLInt: TypeError: filepath.toLowerCase is not a fuction 에러 new webpack.BannerPlugin({ banner: ` Build Date: ${new Date().toLocaleString()} Commit Version: ${childProcess.execSync('git rev-parse --short HEAD')} Author: ${childProcess.execSync('git config user.name')} `,}), 을 추가하니, ESLInt: TypeError: filepath.toLowerCase is not a fuction에러가 잡히는데 혹시 어떻게 처리해야하는지 그리고 무엇때문에 lint에 걸리는지 궁금합니다2. 깃 명령 내용을 홈화면에서 노출시키고 싶은데 어떻게해야하나요? ` Build Date: ${new Date().toLocaleString()} Commit Version: ${childProcess.execSync('git rev-parse --short HEAD')} Author: ${childProcess.execSync('git config user.name')}` 이 내용을 홈화면에 노출하고싶은데 혹시 보통 어떻게 처리하시는지 궁금합니다(js파일에서 const childProcess = require('child_process'); 이렇게 가져오면 child_prcoesschild_process is not defined 에러가 나고 서치한 방법으로는 제대로 해결한걸 못찾아서 질문드립니다)
-
미해결스프링 프레임워크 핵심 기술
질문이 있습니다.
3:53에 XML Configuration File을 생성합니다.저의 경우는 new에 해당 XML Configuration File이 나오질 않았습니다.혹시 이 문제도 유료 버전이 아니여서 그런 것일까요??
-
미해결[리뉴얼] 처음하는 파이썬 데이터 분석 (쉽게! 전처리, pandas, 시각화 전과정 익히기) [데이터분석/과학 Part1]
파일 생성 읽어오기 수정 등
파이썬 입문과 크롤링 수업에서는 openpyxl 라이브러리를 사용해서 excel 파일을 만들거나 읽어오기를 배웠습니다. 현재 하고 있는 이 부분과 차이가 있나요? 라이브러리를 import 하고 안하고와 사용방법을 제외하고 두 방법에 저장 파일 형식의 차이, 효율성 등 비교를 해주실수 있을까요??? 추후 배운 내용에서는 excel 형태는 CSV가 적합한 것으로 이해했는데 데이터 포맷 타입이 CSV이든 , XML 이든 엑셀 파일로 변환이 가능한 것인가요???
-
미해결[리뉴얼] 파이썬입문과 크롤링기초 부트캠프 [파이썬, 웹, 데이터 이해 기본까지] (업데이트)
괄호나 따옴표 자동완성 후 마우스 커서 이동
주피터 노트북은 괄호나 따옴표가 자동완성이 되던데, 내용 입력 후 마우스 커서를 이동시킬때 마우스나 화살표를 사용하는 방법 말고는 없나요? 강사님 예제 설명 하실때 되게 빠르고 자연스럽게 이동하셔서요.
-
미해결스프링 입문 - 코드로 배우는 스프링 부트, 웹 MVC, DB 접근 기술
RowMapper 에 대해 질문 드립니다!!
@Overridepublic List<Member> findAll() { return jdbcTemplate.query("select * from member", memberRowMapper());로우 맵퍼 반환값이 멤버객체 하나를 반환하는데 어떻게 여러 데이터가 담기는지 이해가 잘 안갑니다 ㅠㅠ 반복형태를 도는 코드 숨겨져 있나요 ..?
-
미해결대세는 쿠버네티스 (초급~중급편)
bash: nslookup: command not found 메세지
yum install bind-utils net-tools -y 으로 net-tools 설치후 실행하면 됩니다.
-
미해결모든 개발자를 위한 HTTP 웹 기본 지식
질문합니다
이 강의가 처음에 이해가안갔지만 어느 정도 약간 이해가 가는데요 이게 다른사람 아이피로 Http로 메세지를 전송하는 원리인가요??
-
미해결스프링 입문 - 코드로 배우는 스프링 부트, 웹 MVC, DB 접근 기술
stream() 객체가 궁금해요
아무리 찾아보고 해도 이해가 잘 가지 않는데요 ㅠㅠ filter 나 map 메소드는 알겠는데 stream() 이 뭐 하는건지 감이 안옵니다..
-
미해결프로그래밍 시작하기 : 파이썬 입문 (Inflearn Original)
예제3 차이점을 모르겠습니다.
for c in reader: for k in c.items(): print(k) print(type(k)) print('----------') 하면 k의 타입은 튜플인 것을 볼 수 있습니다. 왜 튜플로 되는지 궁금합니다.
-
미해결프로그래밍 시작하기 : 파이썬 입문 (Inflearn Original)
예제문서 & 예제파일 다운 오류
예제문서&예제파일을 다운받고 열어보면 압축파일에 아무것도 들어있지 않은 상태로 계속 다운로드가 됩니다.. !ㅠㅠ
-
미해결[리뉴얼] React로 NodeBird SNS 만들기
styled-components 적용하고
저장했을때는 잘 적용 되는데 새로고침(f5)을 누르면 왜 적용된게 없어질까요..? 원래이런가요?
-
미해결스프링 부트 개념과 활용
콘솔창 질문
기선님 콘솔창 처럼 하려면 인텔리제이에 플러그인 설치해야되나요? 저는 맥북프로사용중입니다. 혹시 아이맥 전용 콘솔인가요
-
미해결현존 최강 크롤링 기술: Scrapy와 Selenium 정복
xpath와 셀레니움을 활용하여 페이스북 크롤링하기 코드 질문
강의 코드를 그대로 따라했다고 생각했는데 왜 이런 오류가 발생하는지 잘 모르겠네요. login_button의 xpath 코드가 잘못되서 그런 줄 알고 u_0_f로 고쳐서 다시 해봐도 안됬습니다. 뭐가 문제일까요? 감사합니다
-
미해결스프링 핵심 원리 - 기본편
안녕하세요! 스프링 관련해서 질문 드립니다.
안녕하세요 영한님, 항상 좋은 강의 감사드립니다. 수업을 듣던 중 의문점이 생겨서 질문 드립니다. 결국 스프링을 사용하는 목적은 애플리케이션 개발에 필요한 기본 뼈대가 되는 기능들은 제공해 줄테니 개발자들은 비즈니스 로직에 집중하여 생산성을 높이는데 집중하자는 것이 목적이라는 생각이 드는데요. 혹시, 예를들어, 파이썬의 장고나 nodejs는 사용목적이 생산성을 높이기 위해 사용하는 프레임워크가 아닌건가요? 만약, 다른 프레임워크들도 결국 생산성을 높여준다는 것이 목적이라면, 다른 프레임워크들과 비교해서 자바의 스프링이 차별성을 갖는 점은 무엇이길래 메이저 it 기업에서 그토록 많이 사용되는 것인지 궁금합니다.
-
[C#과 유니티로 만드는 MMORPG 게임 개발 시리즈] Part3: 유니티 엔진
crossFade질문
삭제된 글입니다
-
미해결홍정모의 따라하며 배우는 C언어
Printf 함수질문
공부하다가 잠시 의문점을 갖게 된 부분이있습니다,, 구글링을 해봐도 못찾아서 도움을 받으려고 왔습니다 Printf함수에 대해 질문하고싶은데 저가 보는 책(이것이 c언어다)에서, Printf함수가 기본적으로 문자열을 출력하는 함수이고, 정수와 실수를 printf함수로 출력하려면 실제 아스키코드값형태의 문자로 변환시켜야 출력이 가능하고 그래서 정수나 실수를 printf로 찍을때는 변환문자가 필요하고 이 변환 문자를 사용하라고 나와있습니다 Int a=10; Printf(“%d”,a); 하면 원래 a의 대이터 10은 이진수로 00000000 00000000 00000000 00001010 이렇게 저장되있는데 이러한 이진수가 printf (“%d”)로 출력하면 1번째 질문 %d때문에 아스키코드로 변환하는 과정이 나온다음 문자인10이 화면에 출력된다는건가요? 즉 아스키코드가 ‘0’이 48이고 ‘1’이 49인데 메모리에 저장된 이진수들이 %d로 인해서 아스키코드로 변환하는 과정을 거치고 그과정의 결과가 49 48로 나와서 문자열 10이 나오는건가요?? 저가 이해한게 맞나요? ㅠㅠ
-
미해결자바 ORM 표준 JPA 프로그래밍 - 기본편
한 엔티티가 여러가지 쿼리에서 쓰일 때..
안녕하세요 강사님 강의 매번 잘 듣고 있습니다^-^ 듣다가 막연하게 든 생각이 있는데요.. 지금은 Member를 오직 Team 엔티티하고만 조인하는 상황인데, 복잡한 실무로 들어가면, 더 여러가지 엔티티들과 다양하게 조인하는 상황이 있을 것 같은데요. 경험이 적어서 예시가 잘 생각안나지만 상상을 해보면 Member (다) <-> (일) MemberPosition 이라던지 Member (일) <-> (다) MemberDailyReport 이라던지.. Member (일) <-> (일) MemberDetail ? 등등 여러가지 연관관계로 여러 테이블과 여러가지 조인이 있을 것 같습니다. 그런데 이렇게 일부 컬럼에 @ManyToOne이나 @JoinColumn 같은 어노테이션등을 쓰게 되면 하나의 엔티티클래스가, 단 한가지 경우의 조인만 할 수 있게 되는게 아닌가 싶어서요.. 혹시 실무에서는 그런 경우를 어떻게 처리하는지 궁금합니다. 막연하게 든 의문이라.. 장황해서 죄송합니다 ^^;; 감사합니다.