
image
一、本節(jié)目標
前兩章主要講了SpringBoot Kotlin的基本操作,這一章我們將學習使用Kotlin訪問MongoDB,并通過JPA完成(Create,Read,Update,Delete)簡單操作。這里有一個問題什么不選用MySQL數(shù)據(jù)庫呢?
- 答案是 Spring Data Reactive Repositories 目前支持 Mongo、Cassandra、Redis、Couchbase。不支持 MySQL,那究竟為啥呢?那就說明下 JDBC 和 Spring Data 的關系。
Spring Data Reactive Repositories 突出點是 Reactive,即非阻塞的。區(qū)別如下:
- 基于 JDBC 實現(xiàn)的 Spring Data,比如 Spring Data JPA 是阻塞的。原理是基于阻塞 IO 模型 消耗每個調(diào)用數(shù)據(jù)庫的線程(Connection)。
- 事務只能在一個 java.sql.Connection 使用,即一個事務一個操作。
二、構(gòu)建項目及配置
本章不在講解如何構(gòu)建項目了,大家可以參考第一章。這里我們主要引入了mongodb-reactive框架,在pom文件加入下列內(nèi)容即可。
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-mongodb-reactive</artifactId>
</dependency>
如何jar包后我們需要配置一下MongoDB數(shù)據(jù)庫,在application.properties文件中加入一下配置即可,密碼和用戶名需要替換自己的,不然會報錯的。
spring.data.mongodb.host=localhost
spring.data.mongodb.port=27017
spring.data.mongodb.password=student2018.Docker_
spring.data.mongodb.database=student
spring.data.mongodb.username=student
三、創(chuàng)建實體及具體實現(xiàn)
3.1實體創(chuàng)建
package io.intodream.kotlin03.entity
import org.springframework.data.annotation.Id
import org.springframework.data.mongodb.core.mapping.Document
/**
* @description
*
* @author Jwenk
* @copyright intoDream.io 筑夢科技
* @email xmsjgzs@163.com
* @date 2019-03-25,18:05
*/
@Document
class Student {
@Id
var id :String? = null
var name :String? = null
var age :Int? = 0
var gender :String? = null
var sClass :String ?= null
override fun toString(): String {
return ObjectMapper().writeValueAsString(this)
}
}
3.2Repository
package io.intodream.kotlin03.dao
import io.intodream.kotlin03.entity.Student
import org.springframework.data.mongodb.repository.ReactiveMongoRepository
/**
* @description
*
* @author Jwenk
* @copyright intoDream.io 筑夢科技
* @email xmsjgzs@163.com
* @date 2019-03-25,18:04
*/
interface StudentRepository : ReactiveMongoRepository<Student, String>{
}
3.3Service
package io.intodream.kotlin03.service
import io.intodream.kotlin03.entity.Student
import reactor.core.publisher.Flux
import reactor.core.publisher.Mono
/**
* @description
*
* @author Jwenk
* @copyright intoDream.io 筑夢科技
* @email xmsjgzs@163.com
* @date 2019-03-25,18:04
*/
interface StudentService {
/**
* 通過學生編號獲取學生信息
*/
fun find(id : String): Mono<Student>
/**
* 查找所有學生信息
*/
fun list(): Flux<Student>
/**
* 創(chuàng)建一個學生信息
*/
fun create(student: Student): Mono<Student>
/**
* 通過學生編號刪除學生信息
*/
fun delete(id: String): Mono<Void>
}
// 接口實現(xiàn)類
package io.intodream.kotlin03.service.impl
import io.intodream.kotlin03.dao.StudentRepository
import io.intodream.kotlin03.entity.Student
import io.intodream.kotlin03.service.StudentService
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.stereotype.Service
import reactor.core.publisher.Flux
import reactor.core.publisher.Mono
/**
* @description
*
* @author Jwenk
* @copyright intoDream.io 筑夢科技
* @email xmsjgzs@163.com
* @date 2019-03-25,18:23
*/
@Service
class StudentServiceImpl : StudentService{
@Autowired lateinit var studentRepository: StudentRepository
override fun find(id: String): Mono<Student> {
return studentRepository.findById(id)
}
override fun list(): Flux<Student> {
return studentRepository.findAll()
}
override fun create(student: Student): Mono<Student> {
return studentRepository.save(student)
}
override fun delete(id: String): Mono<Void> {
return studentRepository.deleteById(id)
}
}
3.4 Controller實現(xiàn)
package io.intodream.kotlin03.web
import io.intodream.kotlin03.entity.Student
import io.intodream.kotlin03.service.StudentService
import org.slf4j.LoggerFactory
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.web.bind.annotation.*
import reactor.core.publisher.Flux
import reactor.core.publisher.Mono
/**
* @description
*
* @author Jwenk
* @copyright intoDream.io 筑夢科技
* @email xmsjgzs@163.com
* @date 2019-03-25,18:03
*/
@RestController
@RequestMapping("/api/student")
class StudentController {
@Autowired lateinit var studentService: StudentService
val logger = LoggerFactory.getLogger(this.javaClass)
/**
* 保存或新增學生信息
*/
@PostMapping("/")
fun create(@RequestBody student: Student): Mono<Student> {
logger.info("【保存學生信息】請求參數(shù):{}", student)
return studentService.create(student)
}
/**
* 更新學生信息
*/
@PutMapping("/")
fun update(@RequestBody student: Student): Mono<Student> {
logger.info("【更新學生信息】請求參數(shù):{}", student)
return studentService.create(student)
}
/**
* 查找所有學生信息
*/
@GetMapping("/list")
fun listStudent(): Flux<Student> {
return studentService.list()
}
/**
* 通過學生編號查找學生信息
*/
@GetMapping("/id")
fun student(@RequestParam id : String): Mono<Student> {
logger.info("查詢學生編號:{}", id)
return studentService.find(id)
}
/**
* 通過學生編號刪除學生信息
*/
@DeleteMapping("/")
fun delete(@RequestParam id: String): Mono<Void> {
logger.info("刪除學生編號:{}", id)
return studentService.delete(id)
}
}
四、接口測試
這里我們使用Postman來對接口進行測試,關于Postman這里接不用做過多的介紹了,不懂可以自行百度。

image
控制臺打印如下:
2019-03-25 18:57:04.333 INFO 2705 --- [ctor-http-nio-3] i.i.kotlin03.web.StudentController : 【保存學生信息】請求參數(shù):{"id":"1","name":"Tom","age":18,"gender":"Boy","sclass":"First class"}
我們看一下數(shù)據(jù)庫是否存儲了

image
其他接口測試情況

image

image

image

image