Всем привет, меня зовут Олег, я техлид в ДомКлике. В нашей команде ядром стека является Kotlin и Spring Boot. Хочу поделиться с вами своим опытом по взаимодействию и особенностях работы с PostgreSQL и Hibernate в связке со Spring Boot и Kotlin. Также на примере микросервиса, покажу преимущества Kotlin и его отличия от аналогичного приложения на Java. Расскажу о не совсем очевидных сложностях, с которыми могут столкнуться новички при использовании этого стека с Hibernate. Статья будет полезна разработчикам, желающим перейти на Kotlin и знакомых со Spring Boot, Hibernate Java.
Плагины
Для приложения на Kotlin в качестве сборщика проекта возьмём Gradle Kotlin DSL. Список подключенных плагинов будет стандартным для Spring Boot, а для Kotlin с Hibernate у нас появится несколько новых:
plugins {
id("org.springframework.boot") version "2.2.7.RELEASE"
id("io.spring.dependency-management") version "1.0.9.RELEASE"
kotlin("jvm") version "1.3.72"
kotlin("plugin.spring") version "1.3.72"
kotlin("plugin.jpa") version "1.3.72"
}
Рассмотрим три последних.
kotlin(«jvm»)
— базовый плагин Kotlin для работы на JVM. Без которого не заведется ни одно приложение на Java-стеке.kotlin(«plugin.spring»)
— поскольку классы в Kotlin по умолчанию финальны, то этот плагин автоматически сделает классы, помеченные аннотациями @Component
, @Async
, @Transactional
, @Cacheable
и @SpringBootTest
открытыми к наследованию, а в тематике, относящейся этой статье, это позволит классам, написанным на Kotlin быть проксированными в Spring через CGLib прокси.Важно отметить, что сущности, помеченные аннотациями
@Entity
, @MappedSuperclass
и @Embaddable
, не станут open
после подключения плагина. Более того, get accessor
’ы тоже будут финальными, и тогда мы потеряем возможность работать с entity reference
. Чтобы этого избежать и сделать Entity
и его поля open
, добавим в build.gradle.kts:allOpen {
annotation("javax.persistence.Entity")
annotation("javax.persistence.MappedSuperclass")
annotation("javax.persistence.Embeddable")
}
kotlin(«plugin.jpa»)
— Если предыдущие два плагина применяются к любому приложению на Kotlin + Spring Boot, то следующий, уже относится напрямую к Hibernate. А он, как известно, для инициализации Entity
использует рефлексию и инициализирует класс с конструктором без аргументов. Но так как мы пишем на Kotlin, такового конструктора может и не найтись. Если мы определили свой собственный первичный конструктор (primary constructor), то при загрузке Entity
у нас выкинет исключение:org.hibernate.InstantiationException: No default constructor for entity
Зависимости
Набор зависимостей у нас тоже будет не совсем идентичный набору на Java:
dependencies {
implementation("org.springframework.boot:spring-boot-starter-web")
implementation("org.springframework.boot:spring-boot-starter-data-jpa")
implementation("com.fasterxml.jackson.module:jackson-module-kotlin")
implementation("org.jetbrains.kotlin:kotlin-reflect")
implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8")
implementation("org.liquibase:liquibase-core")
runtimeOnly("org.postgresql:postgresql")
testImplementation("org.springframework.boot:spring-boot-starter-test")
testImplementation("org.testcontainers:testcontainers:$testContainersVer")
testImplementation("org.testcontainers:postgresql:$testContainersVer")
}
Добавим еще пару зависимостей в дополнение к стандартному веб-стартеру Spring Boot и к основному интересующему нас стартеру
org.springframework.boot:spring-boot-starter-data-jpa
, который в качестве реализации JPA по умолчанию тянет Hibernate: org.jetbrains.kotlin:kotlin-reflect
— нужен для рефлексии на Kotlin, которая уже поддерживается в Spring Boot и широко используется для инициализации классов.org.jetbrains.kotlin:kotlin-stdlib-jdk8
— добавляет возможность работать с коллекциями Java, поддержку стримов и многое другое. На этом различия в конфигурировании проекта на Kotlin по сравнению с Java у нас заканчиваются, перейдем к самому проекту, его структуре таблиц и сущностей.
Таблицы и сущности
Наше приложение будет состоять из двух таблиц department и employee, которые связаны отношением «один ко многим».
Структура таблиц:
В качестве базы будем использовать СУБД PostgreSQL. Структуру таблиц создадим с помощью liquibase, а в качестве тестовых зависимостей будем использовать стандартный стартер:
org.springframework.boot:spring-boot-starter-test
— тестировать будем в Docker с помощью testcontainers.Сущности
Как и в любом приложении с более чем одной сущностью, создадим общего предка для всех
entity
.BaseEntity
:@MappedSuperclass
abstract class BaseEntity<T> {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
var id: T? = null
override fun equals(other: Any?): Boolean {
other ?: return false
if (this === other) return true
if (javaClass != ProxyUtils.getUserClass(other)) return false
other as BaseEntity<*>
return this.id != null && this.id == other.id
}
override fun hashCode() = 25
override fun toString(): String {
return "${this.javaClass.simpleName}(id=$id)"
}
}
DepartmentEntity
:@Entity
@Table(name = "department")
class DepartmentEntity(
val name: String,
@OneToMany(
mappedBy = "department",
fetch = FetchType.LAZY,
orphanRemoval = true,
cascade = [CascadeType.ALL]
)
val employees: MutableList<EmployeeEntity> = mutableListOf()
) : BaseAuditEntity<Long>() {
fun addEmployee(block: DepartmentEntity.() -> EmployeeEntity) {
employees.add(block())
}
fun setEmployees(block: DepartmentEntity.() -> MutableSet<EmployeeEntity>) {
employees.clear()
employees.addAll(block())
}
}
EmployeeEntity
:@Entity
@Table(name = "employee")
class EmployeeEntity(
val firstName: String,
var lastName: String? = null,
@ManyToOne
@JoinColumn(name = "department_id")
val department: DepartmentEntity
) : BaseAuditEntity<Long>()
Мы не используем Data-классы. Это кажется явным преимуществом Kotlin перед Java (до 14 версии), и этому есть объяснение.
Почему не использовать?
Data-классы, помимо того, что они финальны сами по себе, имеют по всем полям определенные
equals
, hashCode
и toString
. А это недопустимо в связке с Hibernate.Почему? А также зачем
hashCode
всегда равен константе — ответ в документации самого Hibernate. Конкретно нас интересует вот этот раздел: Although using a natural-id is best for equals and hashCode, sometimes you only have the entity identifier that provides a unique constraint.
It’s possible to use the entity identifier for equality check, but it needs a workaround:
- you need to provide a constant value for hashCode so that the hash code value does not change before and after the entity is flushed.
- you need to compare the entity identifier equality only for non-transient entities.
То есть сравнивать нужно либо по
natural id
, либо, как в нашем примере, по primary key id
. Это позволит избежать множества проблем при сравнении сущности и убережет от ее потери при использовании сущности в качестве элемента в Set
. Наличие же
toString
, определенного по всем полям, и вовсе убивает всю ленивость, например, при журналировании сущности, так как будут проинициализированы все поля для вывода в строку.Учитывая особенности Hibernate, эта функциональность Kotlin нам не подойдет.
Конструктор класса
Kotlin позволяет задавать переменным значения через конструктор, чем грех не воспользоваться. Рассмотрим еще раз
DepartmentEntity
:class DepartmentEntity(
val name: String,
@OneToMany(
mappedBy = "department",
fetch = FetchType.LAZY,
orphanRemoval = true,
cascade = [CascadeType.ALL]
)
val employees: MutableList<EmployeeEntity> = mutableListOf()
) : BaseAuditEntity<Long>() {
Также мы можем проинициализировать через конструктор название подразделения, например:
departmentRepository.save(DepartmentEntity(name = "Department One"))
Через конструктор можно инициализировать, в том числе, и список сотрудников employees. Коллекции, разумеется, объявим изменяемыми.
Используйте var/val в зависимости от необходимости изменения поля
Название организации мы пометили как
val
:class DepartmentEntity(
val name: String,
и оно не может быть
null
. Выбор
var
/val
является удобной опцией и зависит от бизнес-логики. Выбирать между var
и val
надо исходя из требования: должно ли поле сущности быть изменяемым.Допустимость null в полях только в соответствии с БД
Насчет допустимости значений
null
в полях всё не так просто. Ранее мы погрузились немного в глубины Hibernate: говоря о plugin.jpa, я упомянул про использование конструктора без аргументов при инициализации сущности.При инициализации полей тоже используется рефлексия. И если в базе в соответствующей колонке хранилось значение
null
, то класс спокойно инициализируется с этим полем со значением null
. При обращении к нему мы рискуем получить NPE, хотя поле и помечено как not nullable
. Чтобы этого не случилось, надо следить за синхронностью структуры таблиц и классов.Если посмотреть на описанное в последних двух разделах более комплексно, то эти правила применимы не только к примитивам, но и к связке сущностей.
Например,
EmployeeEntity
всегда привязан к DepartmentEntity
:class EmployeeEntity(
val firstName: String,
var lastName: String? = null,
@ManyToOne
@JoinColumn(name = "department_id")
val department: DepartmentEntity
) : BaseAuditEntity<Long>()
Department
является не null
и его нельзя изменить, что может избавить от разного рода ошибок, в особенности, если бизнес-логика требует неизменяемости. Репозитории
При использовании Kotlin, у репозиториев из коробки появилась проверка на допустимость
null
. Так, если мы уверены, что при поиске department
по имени результат будет уникальный и единственный, то можно возвращаемый тип указать как non nullable
:interface DepartmentRepository : JpaRepository<DepartmentEntity, Long> {
fun findOneByName(name: String) : DepartmentEntity
}
Здесь
DepartmentEntity
указан единственным и не может быть null
. Если же по какой-то причине мы не нашли искомый department
, то поймаем уже не NPE, а нечто другое:org.springframework.dao.EmptyResultDataAccessException: Result must not be null!
Такая обработка достигается с помощью добавления специализированной поддержки Kotlin в
MethodInvocationValidator
и ReflectionUtils
в spring data commons. lateinit var
Ещё одной фичей Kotlin, которую хотелось бы рассмотреть, является
lateinit var
.Добавим новый класс-предок:
BaseAuditEntity
.@MappedSuperclass
@EntityListeners(AuditingEntityListener::class)
abstract class BaseAuditEntity<T> : BaseEntity<T>() {
@CreatedDate
@Column(updatable = false, nullable = false)
lateinit var created: LocalDateTime
@LastModifiedDate
@Column(nullable = false)
lateinit var modified: LocalDateTime
}
Рассмотрим применение
lateinit var
на примере полей аудита (created
, modified
). lateinit var
— это not null
поле с отложенной инициализацией. Обращение к полю до его инициализации генерирует ошибку:kotlin.UninitializedPropertyAccessException: lateinit property has not been initialized
Как правило, мы обращаемся к полям
created
и modified
уже после того, как сущность была сохранена в БД. В нашем случае, данные в этих поляхпроставляются на этапе сохранения и они not null
, то lateinit var
нам более чем подходит. Итоги
Мы создали приложение, в котором учтены многие преимущества Kotlin, и рассмотрели важные отличия от Java, избежав многих скрытых сюрпризов. Буду рад, если эта статья окажется полезна не только новичкам. Позднее мы продолжим тему общения микросервиса с БД.
Ссылка на приложение.