亮网络解锁器,解锁网络数据的无限可能 了解详情
写点什么

Spring Data 与 MongoDB:不协调的设计

  • 2013-11-14
  • 本文字数:2442 字

    阅读完需:约 8 分钟

MongoDB 是一款非常知名的 NoSQL 文档数据库,而 Spring 则是 Java 领域著名的开源框架。除了构成 Spring 核心的 IoC 与 AOP 之外,Spring 也有大量应用于各个不同领域的子框架,其中 Spring Data 就是专门针对数据处理的一个子项目。在 Spring Data 下有 Spring Data JPA Spring Data MongoDB Spring Data Redis 等子项目,从名字就可以看出来这些子项目所针对的目标。其中,Spring Data MongoDB 是专门针对 MongoDB 的一个子项目,旨在通过 Spring 的方式来操纵 MongoDB,那么这种集成是简化了开发还是阻碍了开发呢?

Prashant Deva Chronon 的创始人与 CTO。他是一位持续创业者,还创办了 Placid Systems、AntlrStudio 及 Virtual Ant。作为一名极客,Deva 喜欢尝试各种新鲜技术,在使用了 Spring Data MongoDB 一段时间后,他认为这个框架在设计上存在着严重的问题,并撰写了文章进行了深入且详尽的分析。

Spring 框架开发者们喜欢到处宣扬他们对 MongoDB 提供的支持,并以此作为优于其他框架的一个资本。不过,如果在真正的项目中使用 Spring Data MongoDB 的话,你就会发现框架在某些特性上的严重缺失以及设计上的巨大失误。

Spring Data MongoDB 试图将 ORM 风格的架构硬塞到非关系型数据库中,这直接造成在实际项目中根本无法使用的严重后果,下面就来谈谈这背后的原因:

无法检索字段,只能获取整个文档

这是 Spring Data MongoDB 设计上最大的瑕疵。它试图用 SQL 数据库中的行对文档进行建模,并且希望你像 ORM 一样创建“实体”类。这两者根本就不是一回事。文档可要比 SQL 数据库中的行复杂多了,也大多了。

对于 SQL 数据库中的行来说,它认为你在大多数查询中都会获取到里面的所有数据,否则数据就应该被划分到多张表中。不过文档与之完全不同,文档可以嵌套,很多时候你只想从数据库中获取到文档的一个子集而已。

但对于 Spring Data MongoDB 的“实体”模型来说,你不得不在每次查询中都检索出整个文档才行。

没有 DBRef 延迟加载

这个就更加疯狂了。如果通过 DBRefs 来引用其他文档,那么 Spring Data MongoDB 就会得到整个文档而不是文档引用。如果通过 DBRefs 连接了大量文档,那么一个只想获得几个字段的简单查询都会导致获取到整个文档图!

实际情况是这个 Bug 报告已经有将近两年时间了,被指定为“低优先级”状态,也没有什么解决方案,这太让人难以置信了,这表明 Spring Data MongoDB 的愿景与实际的使用之间存在着多么大的差距。

不支持游标

想要通过游标遍历集合吗?没门。要么就取出整个集合,要么就转而使用原生的 Mongo Java 驱动。

不完备的聚合框架支持

Spring Data MongoDB 是从不久之前才开始支持 MongoDB 聚合框架的,就像框架的其他部分一样,这种支持也是个半成品。

框架文档很少并且容易让人产生混乱,在实际项目中所需的大多数聚合查询都没法在Spring Data MongoDB 的文档中找到,你只能使用原生的驱动才行。

不完善的索引支持

虽然可以通过框架将某个字段标记为“加索引的”,但其他操作却根本就不知道这回事。

就拿MongoDB 中的一个例子来说吧,如果某个字段是唯一且稀疏的,那么你就不能使用“null”值向文档中插入一次以上。这意味着在第2 次插入时,该字段就不应该在第1 次插入的位置处出现。

然而,由于Spring 强迫你在一个类中定义文档的字段,因此最后需要将null 赋给不存在的字段。如果该字段拥有唯一的稀疏索引,那么这会导致运行期错误,因为Spring 在根据实体对象创建查询时(以注解的形式直接定义在字段上),它根本就不知道索引的存在。

Spring 还提供了一个 ensureIndex() 方法用于手工在字段上创建索引,无需使用注解,不过文档中并没有提及何时该使用这种方式、使用的频率是多少,以及调用的性能是怎样的。

无法切换数据库

很多时候,你希望将数据存储在单个 MongoDB 实例的不同数据库中,比如说要保持不同客户数据的分离。

如果使用原生的 Mongo 驱动,那么切换数据库简直就是小菜一碟,直接调用 getDb(dbName) 就行了。但如果使用 Spring Data MongoDb,那么这几乎是一件无法做到的事情,除非你愿意自己写大量的代码(不过这么做的话在新版框架发布后可能就会出现移植性问题)。

让人奇怪的日志框架

我已经就 Spring 的文档问题专门写过一篇文章。Spring 文档说它使用的是 Jakarta Commons Logging ,不过 Spring Data 显然使用的是 SLF4J。然而,Spring Data 文档却压根儿就没有提过这事。

这意味着如果开始使用 Spring Data,那么你可能就会遇到很多文档中没有提及的运行期错误,这时没别的办法,上 StackOverflow 上找答案吧。

不支持文档的动态特性

使用 MongoDB 这样的面向文档的数据库最大的原因就在于其动态特性了。比如说同一集合中的文档可以不同,这样同一个集合中就可以有多个版本的文档,然后逐渐升级,文档也可以嵌套。键名不必事先知道,这样就可以直接向文档中插入属性映射。

但事实却是 Spring Data MongoDB 丢弃了 MongoDB 的这个最基本的特性,并且试图在其上构建一个确定的、预先定义好的 ORM 风格的层,这直接造成框架与底层数据库之间的不匹配。

结论

Spring Data MongoDB 似乎是由一些压根儿就没有在真正的项目中用过 MongoDB 的人设计出来的,它试图将面向文档的数据库硬塞到 ORM 风格的框架中。

这导致了框架与数据库之间的不匹配,Spring Data MongoDB 反而成为了一个负担而不是帮助我们简化开发。最后,你还是不得不在几乎所有真正的项目开发中使用原生的 MongoDB Java 驱动。

后记

各位 InfoQ 读者,你曾经使用过 Spring Data MongoDB 进行过 MongoDB 开发么,你对于 Spring Data MongoDB 的使用感受是如何呢?是否与 Deva 的经历一致呢?我曾经在项目中尝试过 Spring Data MongoDB,使用下来的一个直观感受就是它将简单的事情搞复杂了。直接使用原生 Java 驱动来操纵 MongoDB 是个非常简单且直观的过程,而 Spring Data MongoDB 却沿用了它一直以来的处理风格,那就是 IoC,需要先进行注入,然后从容器中获取所需的对象,将原本很轻松的操作变得复杂起来,不知各位读者有什么样的感受呢,欢迎讨论。

2013-11-14 10:2411705
用户头像

发布了 88 篇内容, 共 258.8 次阅读, 收获喜欢 8 次。

关注

评论

发布
暂无评论
发现更多内容

Go- 包的使用

HelloBug

Go 语言

公司刚来的京东架构师:看完我写的spring笔记,甩给了我一份文档

编程菌

Java 编程 程序员 计算机 技术宅

蚂蚁金服+拼多多+抖音+天猫Java面经合集,金九银十Java开发校招社招福音!

编程susu

Java 编程 程序员 计算机 技术宅

在线JSON转PHP Array工具

入门小站

工具

Spring的七大模块你了解吗?

4ye

Java spring 架构 后端 8月日更

27岁到来之际,我在阿里实现了年薪40W+的小目标

编程菌

Java 编程 程序员 计算机 技术宅

撒花!!金九银十喜提offer!秋招蚂蚁金服Java研发岗四面

编程菌

Java 编程 程序员 计算机 技术宅

做了好几年Java开发,一直碌碌无为,如今靠着这份面试题跟答案,我从15K变成了30K

编程susu

Java 编程 程序员 计算机 技术宅

实现"双碳"目标,看下纺织业的智慧样本

百度大脑

人工智能

“不服跑个分?” 是噱头还是实力?| 龙蜥技术

OpenAnolis小助手

内核 Cgroups CFS调度器

架构训练营第 1 期 模块六作业

高远

python实现两台不同主机之间进行通信(客户端和服务端)——Socket

Python研究者

8月日更

轮询锁在使用时遇到的问题与解决方案!

王磊

8月日更

SSR 技术概述

编程三昧

SSR 8月日更 服务端渲染

区块链在供应链金融应用优势与四类常见模式

CECBC

centos7中docker安装

消失的子弹

Docker Kubernetes 云原生

06. 第三次AI浪潮:有何不同?

数据与智能

人工智能

C#多线程开发-线程基础 01

Andy阿辉

C# 多线程 8月日更 c#多线程

LeetCode 每日一题「搜索插入位置」

陈皮的JavaLib

Java 面试 算法 LeetCode 8月日更

适女化科技(二):让女性更安全的两条技术路径:软件硬件化与硬件软件化

脑极体

去哪儿网库存搜索在高并发场景下的探索

Qunar技术沙龙

技术 高并发 投票机制

内核热补丁,真的安全么?| 龙蜥技术

OpenAnolis小助手

操作系统 内核 热替换

秒懂消息队列

yuexin_tech

消息队列

Rust从0到1-高级特性-函数和闭包进阶

rust 闭包 函数指针

Go- 包的制作

HelloBug

Go 语言 包的制作

京东三面惨遭被虐,关于redis,高并发,分布式,微服务一窍不通

编程菌

Java 编程 程序员 计算机 技术宅

比特币挖矿的未来只能依靠绿色能源?

CECBC

Linux之tr命令

入门小站

Linux

Vue进阶(八十六):iframe 结合 window.postMessage 实现跨域通信

No Silver Bullet

Vue 8月日更 iframe

netty系列之:轻轻松松搭个支持中文的服务器

程序那些事

Java Netty 程序那些事

微信业务架构&学生管理系统架构选型

John

Spring Data与MongoDB:不协调的设计_DevOps & 平台工程_张龙_InfoQ精选文章