HarmonyOS开发者限时福利来啦!最高10w+现金激励等你拿~ 了解详情
写点什么

我为何停止使用 Spring

  • 2013-12-26
  • 本文字数:2466 字

    阅读完需:约 8 分钟

Johannes Brodwall 是一位程序员、解决方案架构师、用户组与会议组织者、会议演讲者与布道师。Johannes 一直在不遗余力地将敏捷原则应用到大型软件项目中,不过他真正感兴趣的是与全世界的程序员分享更多关于编程的有趣经验。目前,Johannes 就职于 Exilesoft,担任首席科学家一职。近日,Johannes 撰写了一篇关于他为何停止使用Spring 的文章,在程序员群体中引起了很大的反响。

我之前发表的一篇题为谦卑的架构师的文章引起了很多争论,特别是Spring 与依赖注入框架这个话题,这次我打算来谈谈为什么我要停止Spring。我是挪威最早采用Spring 框架的一批人。我们在开发一个大型系统时最后不得不考虑诸如如何重用XML 配置文件的各种不同机制等问题。最后,这演变成了@Autowire 与component-scan,这种方式解决了大量配置文件的难题,不过却降低了人们了解全部源代码的能力,这直接导致开发者被限制在应用中非常小的部分代码中。随着应用变得越来越复杂,文化、工具、文档等东西导致开发者们不得不在一个个不必要的层次上构建另外一些不必要的层次。

不久之后,我尝试在不使用依赖注入框架的情况下来构建应用,不过这引发了另外一些问题,那就是何时该使用“new”呢、何时使用setter 或是构造器参数呢,哪种类型适合作为依赖呢、哪些东西导致了对基础设施的耦合呢。

根据直觉我发现DI 框架确实能够帮助我改善设计,不过与此同时,我还发现在离开容器时,解决方案变得更小(这很棒!)、理解起来更简单,并且易于测试。这让我感到进退维谷,我发现使用容器的代价是非常高的,它会不断增加复杂性与规模,同时会降低一致性。不过话又说回来,它教会了我一些更棒的设计技巧。

总的来说,我个人认为一致、小型的系统要比那些为了解耦而解耦的系统更有价值。一致性与解耦是对立的,我站在一致性这一边。同时,我发现依赖注入文化对重用性有更强的偏爱,不过重用会引入耦合。如果模块A 重用了模块B,那么我们就说模块A 与模块B 是耦合的。模块B 中的变化可能会对模块A 产生更好(修复Bug)或是更糟(引入新的Bug)的影响结果。如果重用所带来的好处更多,那就值得使用;否则就不值得。因此,重用与解耦是对立的两个方面,我自己更偏向于解耦。如果出现冲突,我个人认为优先级应该是一致性大于解耦,解耦大于重用性,而Spring 的基因似乎与此相反。

Johannes 的文章发布后,旋即引起了程序员社区的激烈讨论,有些讨论也很有意思,下面摘录几篇:

Marc Stock 说到:

如果使用 Spring 的依赖注入增加了系统的复杂性,那么问题的症结在于你自己。我使用 Spring 有好几年的经历了,它总是让事情变得更棒和更整洁。我不敢说所有的 Spring 项目都是这样,不过对于依赖注入来说绝对没错。也就是说,我发现有很多使用 Spring 的方式值得商榷,事实上他们做的很多事情都是不必要的(不过他们却并不这么认为)。如果你能举出 Spring 依赖注入会引起混乱的例子,我愿意拭目以待。

Johannes Brodwall 说到:

文章的质疑很不错。我来举个简单的例子,假如一个系统有很多层次,所有东西都被加上了 @Autowire 与 component-scan。我尝试实例化其中的某些服务,不过却缺少依赖。最后只能将所有的依赖加进来来实例化测试中所需的一个简单服务,因为查找存在哪些依赖、应该使用哪些依赖来作为模拟花费了我大量的时间。

Hendy Irawan 说到:

文中提到“同时,我发现依赖注入文化对重用性有更强的偏爱,不过重用会引入耦合。如果模块 A 重用了模块 B,那么我们就说模块 A 与模块 B 是耦合的”。这里需要对“模块”做一些澄清。接口会帮助我们更好地理解。在重用时,使用的是 Spring、CDI 还是你自己的什么东西并不重要。你提到“Spring 文化”,是真的么?举个例子,使用(c3p0)数据源与 PlatformTransactionManager(JpaTransactionManager)来配置一个 JPA EntityManager(FactoryBean)。这里会有大量的重用,不过解耦性却很不错。你可以将 JPA 切换到 Hibernate,也可以将 c3p0 却换到其他数据源,还可以将 TransactionManager 切换到 Hibernate 或是 JTA 的。 你讨厌 XML 配置,也讨厌 @Autowired,不过配置总归是要有的。如果喜欢 set 或是 new 的方式,那么你可以使用注解,文中并没有提及这一点。如果使用的是 CDI,那么讨厌 @Autowired/@Inject 有情可原,不过这是 Spring,你有很多选择。根据我的经验,对于后端服务绑定使用注解配置,对于 UI 组件使用 @Inject 会更好一些。我们也不使用 XML,你可以尝试一下这种方式。

Manuel Rascioni 说到:

这几天我一直在思考是否该使用 Spring,现在我觉得使用是值得的。看看整个(Web)应用,你需要这些东西:一个依赖注入管理系统、一个持久化框架、在各个层之间转换对象的东西、一个安全框架与一个 AOP 系统(管理事务、安全等东西)。你有多种选择,也可以自己创建。如果使用现有的,那需要注意的就是不同框架的集成;如果自己创建,那就需要自己编写大量代码。我的经历告诉我使用 Spring 可以很好地解决这些问题。它不仅仅是个依赖注入框架,还是一套完整的框架生态系统,可以实现组件之间的解耦,同时又很好地实现了这些框架之间的集成。对于测试来说,我使用 mock 框架进行单元测试,而没有使用 Spring(对于单元测试来说它太慢了)。对于集成测试来说,我们需要使用 Spring 配置文件。因此,根据我的经验来看,使用 Spring 是值得的。关于文中提到的耦合,如果模块 C 需要模块 A 的一些东西,你是怎么解决的呢?难道是编写一个新的模块,然后复制模块 A 么?这完全违背了 DRY 原则吧。

各位 InfoQ 读者,你是如何看待文中的观点以及各个评论的看法的呢?Spring 从诞生到现在已经有很多年了,从最早的依赖注入与面向方面编程到现在的一站式框架体系,Spring 本身也变得越来越庞大了,但同时功能也是越来越强大。特别是前不久刚刚发布的 Spring 4.0 更是增加了不少令人激动的新特性,那么在你的项目与系统中是否使用到了 Spring 呢?你觉得 Spring 带给你的好处与它本身的缺陷相比如何呢?换句话说,Spring 框架的性价比高么?欢迎发表评论与大家一同分享你的观点与看法。

2013-12-26 11:2210813
用户头像

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

关注

评论

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

LLVM之父Chris Lattner:为什么我们要重建AI基础设施软件

OneFlow

人工智能 软件系统

容易混淆的基本概念 成员变量 局部变量 全局变量

NewBoy

前端 移动端 iOS 知识体系 7月月更

微信小程序触底加载与下拉刷新的实现

猪痞恶霸

小程序 前端 7月月更

蚁群算法(实例帮助理解)

秃头小苏

蚁群算法 7月月更

非技术部门,如何参与 DevOps?

SoFlu软件机器人

Web3基金会「Grant计划」赋能开发者,盘点四大成功项目

One Block Community

区块链+

ORACLE进阶(三)数据字典详解

No Silver Bullet

oracle 7月月更 数据字典

Node の MongoDB Driver

空城机

mongodb Node 7月月更

SpringBoot Webflux解析

Ethan

RingCentral Android启动优化实践

RingCentral铃盛

android 启动流程

5G NR系统架构

柒号华仔

5G 网络 7月月更

OpenFeign

急需上岸的小谢

7月月更

小红书自研KV存储架构如何实现万亿量级存储与跨云多活

小红书技术REDtech

存储 分布式KV 跨云多活

第五届 Polkadot Hackathon 创业大赛全程回顾,获胜项目揭秘!

One Block Community

区块链 科技

如何用一个插件解决 Serverless 灰度发布难题?

阿里巴巴云原生

阿里云 Serverless 云原生 插件 灰度发布

官宣!第三届云原生编程挑战赛正式启动!

阿里巴巴云原生

阿里云 Serverless 边缘计算 ACK 云原生编程挑战赛

基于昇腾AI丨以萨技术推出视频图像全目标结构化解决方案,达到业界领先水平

科技热闻

不要再说微服务可以解决一切问题了!

博文视点Broadview

Lepton 无损压缩原理及性能分析

vivo互联网技术

对象存储 无损压缩 lepton 图片压缩

Vue.js基础环境的搭建以及简单使用Element-ui

是乃德也是Ned

7月月更

CSS动画篇之炫酷时钟之时钟墙

南城FE

CSS 前端 动画 时钟 7月月更

Qt实现json解析

小肉球

7月月更

线上故障突突突?如何紧急诊断、排查与恢复

阿里巴巴云原生

阿里云 微服务 云原生 故障恢复 诊断

谈谈对Flink框架中容错机制及状态的一致性的理解

百思不得小赵

flink 容错机制 状态 7月月更

LeetCode-145. 二叉树的后序遍历(java)

bug菌

Leet Code 7月月更

基于昇腾AI丨爱笔智能推出银行网点数字化解决方案,实现从总部到网点的信息数字化全覆盖

科技热闻

Java实现单例模式

lambochen

单例模式

【LeetCode】判断矩阵是否是一个 X 矩阵Java题解

Albert

LeetCode 7月月更

基于STM32+华为云IOT设计的智能防盗单车锁

DS小龙哥

7月月更

SAP UI5 ObjectPageLayout 控件使用方法分享

汪子熙

前端开发 Fiori SAP UI5 ui5 7月月更

基于昇腾AI丨高新兴推出城市道路车辆二次识别解决方案,达到业界领先水平

科技热闻

我为何停止使用Spring_语言 & 开发_张龙_InfoQ精选文章