Upgrade to Pro — share decks privately, control downloads, hide ads and more …

Kotlin + Spring Data JPA

VCNC
April 27, 2019

Kotlin + Spring Data JPA

Kotlin에서는 기존 Java 라이브러리를 거의 그대로 사용할 수 있지만 그렇지 않을 때도 있습니다. Kotlin에서 JPA를 사용할 때 Java와의 차이 때문에 겪은 몇가지 문제에 대해 알아보고 해결책을 공유합니다.

VCNC

April 27, 2019
Tweet

More Decks by VCNC

Other Decks in Programming

Transcript

  1. 타다 프로젝트 2018년 6월~ (10개월) Kotlin + Spring Boot +

    JPA + ... Kotlin 코드 약 6만 줄 JPA 엔티티 36종
  2. 목차 1. Kotlin 장점과 Java와의 호환성 2. Spring Data Repository와

    Kotlin 3. JPA Entity와 Kotlin (1) 기본적인 Entity 정의하기 (2) @ManyToOne과 지연 로딩 (3) Kotlin 컬렉션과 @OneToMany
  3. final List<String> greetings = people.stream() .map(it -> "Hello " +

    it.getName() + "!") .collect(Collectors.toList()); val greetings = people.map { "Hello ${it.name}!" } 간결한 코드
  4. Java에서 Kotlin 코드 호출 언제나 자연스럽게 되지는 않음 기존 Java

    코드는 Kotlin에 대해 모름 Java 쪽에서 Kotlin 클래스가 어떻게 ‘보이는’ 지가 중요 특히 리플렉션을 활용하는 경우 (JPA!)
  5. 목차 1. Kotlin 장점과 Java와의 호환성 2. Spring Data Repository와

    Kotlin 3. JPA Entity와 Kotlin (1) 기본적인 Entity 정의하기 (2) @ManyToOne과 지연 로딩 (3) Kotlin 컬렉션과 @OneToMany
  6. Repository (1) public interface UserRepository extends CrudRepository<User, Long> { User

    findByUsername(String username); } interface UserRepository : CrudRepository<User, Long> { fun findByUsername(username: String): User? }
  7. Repository (1) public interface UserRepository extends CrudRepository<User, Long> { User

    findByUsername(String username); } interface UserRepository : CrudRepository<User, Long> { fun findByUsername(username: String): User? }
  8. Repository (1) interface UserRepository : CrudRepository<User, Long> { fun findByUsername(username:

    String): User } personRepository.findByUsername("nobody") org.springframework.dao.EmptyResultDataAccessException: Result must not be null! at org.springframework.data.repository.core.support.MethodInvocationValidator.invoke at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed at org.springframework.aop.framework.JdkDynamicAopProxy.invoke ...
  9. Repository (1) interface UserRepository : CrudRepository<User, Long> { fun findByUsername(username:

    String): User } personRepository.findByUsername("nobody") org.springframework.dao.EmptyResultDataAccessException: Result must not be null! at org.springframework.data.repository.core.support.MethodInvocationValidator.invoke at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed at org.springframework.aop.framework.JdkDynamicAopProxy.invoke https://docs.spring.io/spring-data/jpa/docs/2.1.6.RELEASE/reference/html/#repositories.nullability.kotlin nullability에 주의!
  10. Repository (2) val optionalUser: Optional<User> = userRepository.findById(1) optionalUser.map { it.username

    }.orElse("") optionalUser.orElse(null)?.username ?: "" Java Optional은 Kotlin에서 불편합니다.
  11. findByIdOrNull val optionalUser: Optional<User> = userRepository.findById(1) optionalUser.map { it.username }.orElse("")

    optionalUser.orElse(null)?.username ?: "" val user: User? = userRepository.findByIdOrNull(1) user?.username ?: ""
  12. 목차 1. Kotlin 장점과 Java와의 호환성 2. Spring Data Repository와

    Kotlin 3. JPA Entity와 Kotlin (1) 기본적인 Entity 정의하기 (2) @ManyToOne과 지연 로딩 (3) Kotlin 컬렉션과 @OneToMany
  13. Entity: Java에서 @Entity public class Person { @Id @GeneratedValue private

    Long id; @Column(nullable = false) private String name; // optional private String phoneNumber; // getters, setters } @Entity public class Person { private Long id; private String name; private String phoneNumber; @Id @GeneratedValue public Long getId() { return id; } @Column(nullable = false) public String getName() { return name; } public String getPhoneNumber() { return phoneNumber; } // setters... }
  14. Kotlin으로 바꿔봅시다 @Entity class Person { @Id @GeneratedValue var id:

    Long? = null @Column(nullable = false) var name: String? = null var phoneNumber: String? = null } @Entity class Person { @get:Id @get:GeneratedValue var id: Long? = null @get:Column(nullable = false) var name: String? = null var phoneNumber: String? = null }
  15. Kotlin으로 바꿔봅시다 @Entity class Person { @Id @GeneratedValue var id:

    Long? = null @Column(nullable = false) var name: String? = null var phoneNumber: String? = null } Kotlin property는 명시적 초기화 필요
  16. 좀 더 Kotlin스럽게: Non-nullable Type @Entity class Person { @Id

    @GeneratedValue var id: Long? = null @Column(nullable = false) var name: String = "" var phoneNumber: String? = null } @Entity class Person { @Id @GeneratedValue var id: Long? = null @Column(nullable = false) var name: String? = null var phoneNumber: String? = null }
  17. 좀 더 Kotlin스럽게: Named Arguments val person = Person( id

    = 1, name = "hi", phoneNumber = "1234" ) val person = Person() person.id = 1 person.name = "hi" person.phoneNumber = "1234"
  18. @Entity class Person { @Id @GeneratedValue var id: Long? =

    null @Column(nullable = false) var name: String = "" var phoneNumber: String? = null constructor() constructor(id: Long?, name: String, phoneNumber: String?) { this.id = id this.name = name this.phoneNumber = phoneNumber } }
  19. 좀 더 Kotlin스럽게: Primary Constructor @Entity class Person( @Id @GeneratedValue

    var id: Long? = null, @Column(nullable = false) var name: String = "", var phoneNumber: String? = null ) @Entity class Person { @Id @GeneratedValue var id: Long? = null @Column(nullable = false) var name: String = "" var phoneNumber: String? = null constructor() constructor(id: Long?, name: String, phoneNumber: String?) { this.id = id this.name = name this.phoneNumber = phoneNumber } }
  20. 좀 더 Kotlin스럽게: 기본값 없애기 @Entity class Person( @Id @GeneratedValue

    var id: Long?, @Column(nullable = false) var name: String, var phoneNumber: String? ) @Entity class Person( @Id @GeneratedValue var id: Long? = null, @Column(nullable = false) var name: String = "", var phoneNumber: String? = null )
  21. 좀 더 Kotlin스럽게: 기본값 없애기 @Entity class Person( @Id @GeneratedValue

    var id: Long?, @Column(nullable = false) var name: String, var phoneNumber: String? ) @Entity class Person( @Id @GeneratedValue var id: Long? = null, @Column(nullable = false) var name: String = "", var phoneNumber: String? = null ) org.springframework.orm.jpa.JpaSystemException: No default constructor for entity: : com.example.demo.Person
  22. kotlin-noarg (kotlin-jpa) Kotlin 컴파일러 플러그인 ✨ 특정 어노테이션이 붙은 클래스에

    no-arg constructor를 자동으로 만들어줍니다. Gradle/Maven 플러그인으로 추가합니다. kotlin-jpa kotlin-noarg + JPA를 위한 기본 설정 @Entity, @Embeddable, @MappedSuperclass http://kotlinlang.org/docs/reference/compiler-plugins.html#jpa-support
  23. @Entity class Person( @Id @GeneratedValue var id: Long?, @Column(nullable =

    false) var name: String, var phoneNumber: String? ) feat. kotlin-jpa 플러그인
  24. Data class @Entity data class Person( @Id @GeneratedValue var id:

    Long?, @Column(nullable = false) var name: String, var phoneNumber: String? )
  25. Data class @Entity data class Person( @Id @GeneratedValue var id:

    Long?, @Column(nullable = false) var name: String, var phoneNumber: String? ) equals/hashCode를 호출하게 될 때 주의 필요 ==, toSet() 순환 참조가 있으면 무한 재귀 호출에 빠짐
  26. 목차 1. Kotlin 장점과 Java와의 호환성 2. Spring Data Repository와

    Kotlin 3. JPA Entity와 Kotlin (1) 기본적인 Entity 정의하기 (2) @ManyToOne과 지연 로딩 (3) Kotlin 컬렉션과 @OneToMany
  27. Asset @ManyToOne과 지연 로딩 @Entity class Asset( @Id var id:

    Long?, @Column(nullable = false) var name: String, @ManyToOne(fetch = FetchType.LAZY) var person: Person ) Person Asset Asset
  28. 지연 로딩: 기대 val a = assetRepository.findByIdOrNull(1)!! SQL: select ...

    from asset asset0_ where asset0_.id=? Person에 대한 쿼리 X
  29. 지연 로딩: 기대 val a = assetRepository.findByIdOrNull(1)!! val person =

    a.person println(person::class) class com.example.demo.Person$HibernateProxy$lsHeDMGk 프록시 객체 Person에 대한 쿼리 X
  30. 지연 로딩: 기대 val a = assetRepository.findByIdOrNull(1)!! val person =

    a.person println(person.id) Person에 대한 쿼리 X (Asset만으로 알 수 있음)
  31. 지연 로딩: 기대 val a = assetRepository.findByIdOrNull(1)!! val person =

    a.person println(person.id) println(person.name) SQL: select ... from person person0_ where person0_.id=? 지연 로딩
  32. 지연 로딩: 실제 val a = assetRepository.findByIdOrNull(1)!! SQL: select ...

    from asset asset0_ where asset0_.id=? SQL: select ... from person person0_ where person0_.id=?
  33. 지연 로딩: 실제 val a = assetRepository.findByIdOrNull(1)!! val person =

    a.person println(person::class) SQL: select ... from asset asset0_ where asset0_.id=? SQL: select ... from person person0_ where person0_.id=? class com.example.demo.Person 프록시 객체가 아니다?
  34. Final by default Kotlin 클래스는 final (상속 불가)이 기본 프록시

    클래스를 생성하려면 클래스가 상속 가능해야 합니다.
  35. open! open! open! @Entity class Person( @Id @GeneratedValue var id:

    Long?, @Column(nullable = false) var name: String, var phoneNumber: String? ) @Entity open class Person( @Id @GeneratedValue open var id: Long?, @Column(nullable = false) open var name: String, open var phoneNumber: String? )
  36. kotlin-allopen Kotlin 컴파일러 플러그인 ✨ 특정 어노테이션이 붙은 클래스와 그

    클래스의 멤버를 자동으로 open으로 만들어줍니다. Gradle/Maven 플러그인으로 추가합니다. http://kotlinlang.org/docs/reference/compiler-plugins.html#all-open-compiler-plugin allOpen { annotation("javax.persistence.Entity") }
  37. kotlin-allopen @Entity class Person( @Id @GeneratedValue var id: Long?, @Column(nullable

    = false) var name: String, var phoneNumber: String? ) @Entity open class Person( @Id @GeneratedValue open var id: Long?, @Column(nullable = false) open var name: String, open var phoneNumber: String? ) ✨
  38. 목차 1. Kotlin 장점과 Java와의 호환성 2. Spring Data Repository와

    Kotlin 3. JPA Entity와 Kotlin (1) 기본적인 Entity 정의하기 (2) @ManyToOne과 지연 로딩 (3) Kotlin 컬렉션과 @OneToMany
  39. Kotlin 컬렉션 Immutable Mutable Java List<T> MutableList<T> java.util. List<T> Set<T>

    MutableSet<T> java.util. Set<T> Map<K, V> MutableMap<K, V> java.util. Map<K, V>
  40. @OneToMany @Entity class Person( ..., @OneToMany var assets: List<Asset> )

    org.hibernate.AnnotationException: Collection has neither generic type or OneToMany.targetEntity() defined: com.example.demo.Person.assets
  41. Type Variance Kotlin Java List<out T> java.util.List <? extends T>

    Set<out T> java.util.Set <? extends T> Map<K, out V> java.util.Map <K, ? extends V> https://kotlinlang.org/docs/reference/java-to-kotlin-interop.html#variant-generics
  42. 정리 똑같이 생겼지만 의미가 다른 코드를 조심하자 Java Kotlin T

    nullable non-nullable class non-final final List<T> mutable immutable
  43. 정리 간결한 코드를 위해서 컴파일러 플러그인의 도움을 받을 수 있다

    kotlin-jpa, kotlin-allopen Java 호환성 문제가 있을 때는 바이트코드를 확인해보자