Kotlin与MongoDB整合CURD案例详解

1、mongodb的低版本bson无法转换类型

  • 比如MongoDB数据库表的字段类型为Decimal,实体类用String去定义就会报如下错误
  • No converter found capablof converting from type [org.bson.types.Decimal128] to type [java.lang.String]

解决方法

2、排除自带的MongoDB依赖,添加自定义添加依赖

  • spring-boot-starter-data-mongodb有多少个与MongoDB有关系的都排除出去
<dependency>
          <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-mongodb</artifactId>
            <!-- 如果日志使用log4j2,需要排除,-->
            <exclusions>
                <exclusion>
                    <artifactId>logback-classic</artifactId>
                    <groupId>ch.qos.logback</groupId>
                </exclusion>
                <exclusion>
                    <artifactId>spring-boot-starter-logging</artifactId>
                    <groupId>org.springframework.boot</groupId>
                </exclusion>
                <!--排除低版本的MongoDB驱动-->
                <exclusion>
                    <groupId>org.mongodb</groupId>
                    <artifactId>mongodb-driver</artifactId>
                </exclusion>
                <exclusion>
                    <groupId>org.mongodb</groupId>
                    <artifactId>mongodb-driver-core</artifactId>
                </exclusion>
                <exclusion>
                    <groupId>org.mongodb</groupId>
                    <artifactId>bson</artifactId>
                </exclusion>
            </exclusions>
        </dependency>

3、自定义高版本的jar依赖

<dependency>
    <groupId>org.mongodb</groupId>
    <artifactId>mongodb-driver</artifactId>
    <version>3.12.5</version>
</dependency>
<dependency>
    <groupId>org.mongodb</groupId>
    <artifactId>mongodb-driver-core</artifactId>
    <version>3.12.5</version>
</dependency>

<dependency>
    <groupId>org.mongodb</groupId>
    <artifactId>bson</artifactId>
    <version>3.12.5</version>
</dependency>

4、数据源配置解决方法

  • MongoDB报Exception authenticating MongoCredential异常及权限问题处理解决,参考文章处理
  • mongodb://用户名:密码@机器IP:端口/数据库名?授权源=授权的名称
  • 由于安装的时候是以授权模式进行安装的,详情请参考Docker安装MongoDB
# MongoDB数据库配置
# mongodb://用户名:密码@机器IP:端口/数据库名?授权源=授权的名称
spring.data.mongodb.uri=mongodb://liangjl:123456@192.168.1.231:27017/mongoDB?authSource=admin

5、实体类


import org.springframework.data.annotation.Id
import org.springframework.data.mongodb.core.mapping.Document
import java.math.BigDecimal
import java.util.*

@Document(collection = "sales")
data class Sales constructor(
        /**
         * id
         */
        @Id
        var id: String? = null,
        /**
         * 条目
         */
        var item: String? = null,
        /**
         * 价钱
         */
        var price: BigDecimal? = null,
        /**
         * 数量
         */
        var quantity: Int? = null,
        /**
         * 时间
         */
        var date: Date? = null
)

6、SalesService详解

  • 查询所有销售订单数据
@Autowired lateinit var mongoTemplate: MongoTemplate

/**
 * 查询所有销售订单数据
 * @return
 */
fun findAll(): List<Sales> {
    return mongoTemplate.findAll(Sales::class.java)
}
  • 查询所有销售订单数据,通过价格进行排序

/**
 * 查询所有销售订单数据,通过价格进行排序
 * SQL 语句中, asc是指定列按升序排列,desc则是指定列按降序排列。
 * @return
 */
fun findSort(): List<Sales> {
    //按照价格与数量字段升序查询出来,
    var sort = Sort(Sort.Direction.ASC, "price").and(Sort(Sort.Direction.ASC, "quantity"))
     //查询item为abc的数据进行
    var criteria = Criteria.where("item").`is`("abc")
    var query = Query(criteria)
    //限制1000条数据
    return mongoTemplate.find(query.with(sort).limit(1000), Sales::class.java)
}
  • 查询所有销售订单数据,通过价格进行排序
    /**
     * 查询所有销售订单数据,通过价格进行排序
     * SQL 语句中, asc是指定列按升序排列,desc则是指定列按降序排列。
     * @return
     */
    fun findSort(): List<Sales> {
        //按照价格与数量字段升序查询出来,
        var sort = Sort(Sort.Direction.ASC, "price").and(Sort(Sort.Direction.ASC, "quantity"))
         //查询item为abc的数据进行
        var criteria = Criteria.where("item").`is`("abc")
        var query = Query(criteria)
        //限制1000条数据
        return mongoTemplate.find(query.with(sort).limit(1000), Sales::class.java)
    }
  • gt 大于函数的使用,查询价格大于6元的销售订单数据
    fun findGt(): List<Sales> {
        //查询价格大于6元的销售订单数据
        var criteria = Criteria.where("price").gt(6)
        var query = Query(criteria)
        return mongoTemplate.find(query, Sales::class.java)
    }
  • 查询价格小于6元的销售订单数据
    fun findLt(): List<Sales> {
        //查询价格小于6元的销售订单数据
        var criteria = Criteria.where("price").lt(6)
        var query = Query(criteria)
        return mongoTemplate.find(query, Sales::class.java)
    }

  • 通过Id去查询销售订单数据
    /**
    * 在MongoDB插入脚本的时候是以数字的进行插入是,这里用String去根据Id去查询是查行不到数据
    * 在mongoDB查询是数据变成,所以id可以看做Long或者Double类型入参进行查询
    *
    * fun getOne(id: Long): Sales? {}
    * fun getOne(id: Double): Sales? {}
    *
    * {
    *    "_id" : 1.0,
    *    "item" : "abc",
    *    "price" : NumberDecimal("10"),
    *    "quantity" : 2,
    *    "date" : ISODate("2014-03-01T08:00:00.000Z")
    * }
    */
    fun getOne(id: Long): Sales? {
       val query = Query(Criteria.where("id").`is`(id))
       var sales = mongoTemplate.findOne(query, Sales::class.java)
       return sales
    }
  • 插入数据
    /**
    * 插入数据
    */
    fun addSales() {
       var id = IdWorker.getId().toString()
       var sale = Sales(id, "葡萄", BigDecimal(20), 1, Date())
       //方法一
       //mongoTemplate.insert(sale)
       //方法二
       mongoTemplate.save(sale)
    }
  • 删除数据
/**
 * 删除数据
 */
fun deleteById(id: String) {
    val query = Query(Criteria.where("id").`is`(id))
    mongoTemplate.remove(query, Sales::class.java)
}
  • 修改数据
/**
 * 修改数据
 */
fun update(sales: Sales) {
    var query = Query(Criteria.where("id").`is`(sales.id))
    var update = Update().set("item", sales.item).set("quantity", sales.quantity)
    //更新查询返回结果集的第一条
    var first = mongoTemplate.updateFirst(query, update, Sales::class.java)

    println("update:" + first)

    //更新查询返回结果集的所有
    // mongoTemplate.updateMulti(query,update, Sales::class.java)
}

7、SalesController代码


@RestController
@RequestMapping("/rest/sales/")
open class SalesController : BaseController() {

    @Autowired lateinit var salesService: SalesService

    /**
     * 查询所有信息
     */
    @GetMapping("/findAll")
    fun findAll(): List<Sales>? {
        return salesService.findAll()
    }

    /**
     * 排序的使用
     */
    @GetMapping("/findSort")
    fun findSort(): List<Sales>? {
        return salesService.findSort()
    }


    /**
     * 大于的使用
     */
    @GetMapping("/findGt")
    fun findGt(): List<Sales>? {
        return salesService.findGt()
    }
    /**
     * 小于的使用
     */
    @GetMapping("/findLt")
    fun findLt(): List<Sales>? {
        return salesService.findLt()
    }


    /**
     * 根据Id去查询一条数据
     */
    @GetMapping("/getOne/{id}")
    fun getOne(@PathVariable("id") id:Long): Sales? {
        return salesService.getOne(id)
    }


    /**
     * 插入一条数据
     */
    @PostMapping("/insert")
    fun insert(): Unit {
        return salesService.addSales()
    }
    /**
     * 删除一条数据
     */
    @PutMapping("/deleteById/{id}")
    fun deleteById(@PathVariable("id") id:String): Unit {
        return salesService.deleteById(id)
    }


    /**
     * 修改一条数据
     * 假设如下的数据,传的JSON体为
     * Content-Type = application/json;charset=UTF-8
     * 
     * {  "id" : "1278573453263409154",  "item" : "苹果",  "quantity" : 10 }
     */
    @PutMapping("/update")
    fun update(@RequestBody sales:Sales): Unit {
        return salesService.update(sales)
    }

}

8、数据库表脚本

db.sales.insertMany([
  { "_id" : 1, "item" : "abc", "price" : NumberDecimal("10"), "quantity" : NumberInt("2"), "date" : ISODate("2014-03-01T08:00:00Z") },
  { "_id" : 2, "item" : "jkl", "price" : NumberDecimal("20"), "quantity" : NumberInt("1"), "date" : ISODate("2014-03-01T09:00:00Z") },
  { "_id" : 3, "item" : "xyz", "price" : NumberDecimal("5"), "quantity" : NumberInt( "10"), "date" : ISODate("2014-03-15T09:00:00Z") },
  { "_id" : 4, "item" : "xyz", "price" : NumberDecimal("5"), "quantity" :  NumberInt("20") , "date" : ISODate("2014-04-04T11:21:39.736Z") },
  { "_id" : 5, "item" : "abc", "price" : NumberDecimal("10"), "quantity" : NumberInt("10") , "date" : ISODate("2014-04-04T21:23:13.331Z") },
  { "_id" : 6, "item" : "def", "price" : NumberDecimal("7.5"), "quantity": NumberInt("5" ) , "date" : ISODate("2015-06-04T05:08:13Z") },
  { "_id" : 7, "item" : "def", "price" : NumberDecimal("7.5"), "quantity": NumberInt("10") , "date" : ISODate("2015-09-10T08:43:00Z") },
  { "_id" : 8, "item" : "abc", "price" : NumberDecimal("10"), "quantity" : NumberInt("5" ) , "date" : ISODate("2016-02-06T20:20:13Z") },
])

9、运行结果

[查询全部销售订单数据](http://localhost:8080/rest/sales/findAll)

[根据Id查询销售订单](http://localhost:8080/rest/sales/getOne/1)

  • 更新
  • 查询全部销售订单数据
  • 根据Id去查询

10、Docker安装MongoDB

11、整合原代码Github地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,425评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,058评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,186评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,848评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,249评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,554评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,830评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,536评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,239评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,505评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,004评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,346评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,999评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,060评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,821评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,574评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,480评论 2 267