resources:template/ + {ViewName} + .html 경로를 통해 스프링 부트에 thymeleaf viewName이 매핑된다. 기본 정적 화면은 /static/index.html에 정의한다.
spring-boot-devtools 라이브러리를 추가한 뒤 html 파일을 컴파일만 하면 서버 재시작 없이 변경 사항이 반영된다. build-recompile 메뉴에서 컴파일 할 수 있다.
H2 데이터베이스 설치
개발이나 테스트 용도로 사용할 수 있는 가볍고 편리한 데이터베이스다. 웹 화면을 제공한다.
spring:datasource:url:jdbc:h2:tcp://localhost/~/jpashopusername:sapassword:driver-class-name:org.h2.Driverjpa:hibernate:ddl-auto:create# 애플리케이션 실행 시점에 테이블을 drop하고 다시 생성한다.properties:hibernate:# show_sql: true # System.out에 하이버네이트 실행 SQL을 남긴다.format_sql:truelogging.level:org.hibernate.SQL:debug# logger를 통해 하이버네이트 실행 SQL을 남긴다.
운영 환경의 모든 로그는 가급적 로거를 통해 남겨야 한다. show_sql은 System.out에서 찍기 때문에 가급적 안 쓰는 게 좋다.
application.yml같은 yml 파일은 띄어쓰기 2칸으로 계층을 만든다. 따라서 반드시 맞춰줘야 한다.
@RepositorypublicclassMemberRepository {// 스프링 컨테이너가 EntityManager를 주입할 수 있게 해주는 애너테이션 @PersistenceContextprivateEntityManager em;publicLongsave(Member member) {em.persist(member);// 커맨드와 쿼리를 분리하자.// 저장은 사이드 이펙트를 일으키는 커맨드 성이기 때문에 리턴을 안 하거나 아이디 정도만 반환한다.returnmember.getId(); }publicMemberfind(Long id) {returnem.find(Member.class, id); }}
// 스프링과 관련된 테스트를 할 것이라고 알려준다.@RunWith(SpringRunner.class)// 스프링 부트 프로젝트이므로 함께 넣어줘야 한다.@SpringBootTestpublicclassMemberRepositoryTest { @AutowiredMemberRepository memberRepository; @Test// 추가하지 않으면 // No EntityManager with actual transaction available for current thread// - cannot reliably process 'persist' call 에러가 발생한다.// 엔티티의 데이터 변경은 모두 트랜잭션 안에서 이루어져야 하는데 트랜잭션이 없어서 나는 에러다.// @Transactional은 테스트에 있으면 테스트가 끝난 뒤 바로 롤백한다. @Transactional// 테스트 후 데이터를 날리고 싶지 않으면 이 옵션을 추가한다. @Rollback(value =false)publicvoidsave() {// givenMember member =newMember();member.setUsername("memberA");// whenLong savedId =memberRepository.save(member);Member findMember =memberRepository.find(savedId);// thenAssertions.assertThat(findMember.getId()).isEqualTo(member.getId());Assertions.assertThat(findMember.getUsername()).isEqualTo(member.getUsername());// 같은 트랜잭션에 묶여있기 때문에 같은 영속성 컨텍스트에 존재한다.// 같은 영속성 컨텍스트에서는 아이디 값이 같으면 같은 Entity로 식별한다.// 이미 같은 영속성 컨텍스트에서 관리되고 있는 Entity가 있기 때문에 1차 캐시에서 가져온다.// 따라서 둘을 비교하면 true가 나온다.Assertions.assertThat(findMember).isEqualTo(member); } @Testpublicvoidfind() { }}