2025上半年,最新 AI实践都在这!20+ 应用案例,任听一场议题就值回票价 了解详情
写点什么

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:2411864
用户头像

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

关注

评论

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

深化合作!港华集团数智升级,构建一体化管控平台

用友BIP

什么是零信任

天翼云开发者社区

零信任 SDP架构

WebGL软件开发的技术方案

北京木奇移动技术有限公司

软件外包公司 webgl开发 webgl技术

局域网聊天工具,哪款更适合政企的即时通讯需求?

BeeWorks

即时通讯 IM 私有化部署 局域网

5月23日开课!“北大-用友CIO/CDO数智化进阶课程”

用友BIP

1688 商品数据接口终极指南:Python 开发者如何高效获取标题 / 价格 / 销量数据(附调试工具推荐)

tbapi

1688商品列表接口 1688API 1688商品数据采集

AI应用及数据驱动型HR决策呈激增态势

财见

Addepar完成2.3亿美元G轮融资

财见

0.4元/TB/月!天翼云HBlock打响软件定义存储价格战

天翼云开发者社区

存储 天翼云HBlock

产品经理用什么软件做竞品分析?AI生成产品报告攻略奉上!

职场工具箱

AI 产品经理 竞品分析 办公软件 AIGC

模块化和DID能让UBI惠及所有人吗?

BlockChain先知

高途公布2025年第一季度未经审计业绩及新股票回购计划

财见

利用 Amazon Bedrock Data Automation(BDA)对视频数据进行自动化处理与检索

亚马逊云科技 (Amazon Web Services)

区块链RWA软件系统技术方案

北京木奇移动技术有限公司

区块链技术 软件外包公司 RWA开发

区块链ETF软件系统的技术方案

北京木奇移动技术有限公司

区块链技术 软件外包公司 区块链ETF

区块链RWA系统开发框架

北京木奇移动技术有限公司

区块链技术 软件外包公司 RWA开发

针对大事务问题对业务存储过程改造

GreatSQL

深入剖析低代码:实现24小时产品上线的技术机制与优化策略

JeeLowCode低代码平台

低代码 低代码平台 低代码前端 低代码缺点 低代码,

哈尔滨等保、备案单位如何选择备案地?

黑龙江陆陆信息测评部

内网im聊天软件,私有化部署安全可控

BeeWorks

即时通讯 IM 私有化部署

向日葵、ToDesk、网易UU远程、RayLink,哪款远控软件最适合设计师?

科技热闻

MCP与华为云CSE珠联璧合,打造AI时代微服务生态引擎

华为云开发者联盟

微服务 华为云开发者联盟 MCP 华为云CSE

葵花药业集团与用友战略签约,共启医药行业数智化新篇章

用友BIP

网站进行IPv6改造的步骤有哪些?一文看懂

国科云

技术贴 | 如何基于 KWDB 快速实现工业物联网智能电表系统设计?

KaiwuDB

数据库·

模块化和DID能让UBI惠及所有人吗?

石头财经

第五章:数据治理启动的契机

数据小吏

数据治理

“全球金牌敏捷课程” · 7月19-20日CSM认证课程 · Jim老师引导讨论AI & Agility话题

ShineScrum

敏捷 敏捷教练 CSM认证 CSM认证培训 敏捷认证

【AI智能助手】轻松打造智能助手,定制专属个性风格

JEECG低代码

AI大模型 AI应用 AIGC AI智能助手

Blender 入门教程(四):动画制作

北桥苏

游戏引擎 blender CocosCreator

0元打造520浪漫神器,用CodeBuddy生成专属爱情故事书

不惑

CodeBuddy

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