使用 Spring Data CrudRepository 进行数据库访问
在这一章中, 你将会迁移服务层, 使用 Spring Data CrudRepository
进行数据库访问, 而不是原来的 JdbcTemplate
. CrudRepository 是一个 Spring Data 接口, 可以指定类型的仓库进行通常的 CRUD 操作. 它提供了一些现成的方法来操作数据库.
更新你的应用程序
首先, 你需要调整 Message
类, 来配合 CrudRepository
API:
向
Message
类添加@Table
注解, 声明它与数据库表的映射关系. 在id
属性之前添加@Id
注解.import org.springframework.data.annotation.Id import org.springframework.data.relational.core.mapping.Table @Table("MESSAGES") data class Message(@Id var id: String?, val text: String)除了添加这些注解之外, 你还需要让
id
成为可变属性 (var
), 因为在将新对象插入到数据库时CrudRepository
需要如此.为
CrudRepository
声明一个接口, 它负责操作Message
数据类:import org.springframework.data.repository.CrudRepository interface MessageRepository : CrudRepository<Message, String>更新
MessageService
类. 它现在调用MessageRepository
, 而不是执行 SQL 查询:import java.util.* @Service class MessageService(val db: MessageRepository) { fun findMessages(): List<Message> = db.findAll().toList() fun findMessageById(id: String): List<Message> = db.findById(id).toList() fun save(message: Message) { db.save(message) } fun <T : Any> Optional<out T>.toList(): List<T> = if (isPresent) listOf(get()) else emptyList() }- 扩展函数
CrudRepository
接口中findById()
函数的返回类型是Optional
类的实例. 但是, 从代码统一的角度来说, 返回一个包含单个 message 的List
会更加方便. 要做到这一点, 如果Optional
中有值, 你需要解包这个值, 并返回一个包含这个值的 List. 这部分功能可以实现为Optional
类型的一个 扩展函数.在上面的代码中,
Optional<out T>.toList()
,.toList()
是Optional
的扩展函数. 使用扩展函数, 你可以向任何类添加额外的函数, 当你想要扩展某些库中的类的功能时, 这样会非常有用.- CrudRepository save() 函数
这个函数的工作方式 是假定新的对象在数据库中没有 id. 因此, 对 insertion 操作, id 需要为 null.
如果 id 不是 null,
CrudRepository
假定对象在数据库中已经存在, 并且这是一个 update 操作, 而不是 insert 操作. 在 insert 操作之后,id
会由数据库生成, 并反过来赋值给Message
实例. 这就是id
属性需要使用var
关键字来声明的原因.
更新 messages 表定义, 对 insert 的对象生成 id. 由于
id
是一个字符串, 你可以使用RANDOM_UUID()
函数来生成默认的 id 值:CREATE TABLE IF NOT EXISTS messages ( id VARCHAR(60) DEFAULT RANDOM_UUID() PRIMARY KEY, text VARCHAR NOT NULL );更新
src/main/resources
文件夹中的application.properties
文件内的数据库名称:spring.datasource.driver-class-name=org.h2.Driver spring.datasource.url=jdbc:h2:file:./data/testdb2 spring.datasource.username=name spring.datasource.password=password spring.sql.init.schema-locations=classpath:schema.sql spring.sql.init.mode=always
下面是 DemoApplication.kt
的完整代码:
运行应用程序
应用程序可以在此运行了. 通过将 JdbcTemplate
替换为 CrudRepository
, 功能并没有变更, 因此应用程序应该和以前相同的方式运行.
下一步做什么
得到你个人的语言导航地图, 它可以帮助你浏览 Kotlin 的功能特性, 并追踪你学习语言的进度:
学习如何使用 Java 到 Kotlin 转换器 将既有的 Java 代码转换为 Kotlin.
阅读我们的 Java 代码向 Kotlin 迁移指南: