空调吸气和排气_吸气剂和二传手被认为有害

空调吸气和排气

Java程序员习惯性地用“ getters”和“ setters”来修饰类,这种做法根深蒂固,以至于几乎没有人质疑为什么这样做或是否应该这样做。 最近,我认为最好不要这样做,并且我开始在编写的Java代码中避免使用它。 在这篇博客文章中,我将解释原因。 但是首先,要进行快速的历史课程。

JavaBeans

Getters和setters起源于JavaBeans规范 ,该规范最初于1996年末发布,并于1997年8月更新为1.01版。最初的想法是使对象的创建成为可能,这些对象可以用作构建模块来组成应用程序。 这个想法过去了,“用户”可以使用某种构建器工具连接在一起,并自定义一组JavaBeans组件以一起充当应用程序。 例如,AWT应用程序中的按钮将是Bean(AWT是Java UI库Swing的前身)。 另外,一些JavaBeans更像是常规应用程序,然后可以将它们组合成复合文档,因此电子表格bean可以嵌入到网页内。

当遵循以下约定时,对象就是JavaBean:

  1. 它必须具有一个零参数的构造函数,该构造函数不能失败。
  2. 它具有可通过“ getter”和“ setter”方法访问和更改的属性。
  3. 对于称为Foo的bean的任何属性,访问器方法必须称为getFoo 。 在布尔属性的情况下,可以将吸气剂称为isFoo
  4. Foo的setter方法必须称为setFoo
  5. Bean不必为每个属性都同时提供一个getter和一个setter:一个具有getter而没有setter的属性是只读的; 具有setter且没有getter的属性是只写的。

该规范描述了许多不同的用例,但是从上面的描述可以清楚地看出,JavaBeans被视为具有行为的对象,而不仅仅是数据包。 这个想法已经淡出人们的视线,但是尽管JavaBean被很大程度上遗忘了,但是Java中的getter和setter方法的惯用法仍然存在。

隐喻是错误的

“获取”和“设置”的概念似乎很自然,但这是否正确? JavaBeans约定使用“ get”来表示查询,这是一个没有副作用的操作,但在现实世界中,getting是一个会改变状态的动作:如果我将一本书下架,则该书不再上架。 您可能会反对,这只是纯粹的学问,但我认为这种误解会鼓励我们错误地思考我们编写对象以进行交互的方式。 例如,如果我们有一个Thermometer类,那么大多数Java开发人员都会编写代码来读取温度,如下所示:

Temperature t = thermometer.getTemperature();

确实,“获取”温度是温度计的工作吗? 没有! 温度计的作用是测量温度。 我为什么要为此努力呢? 这是因为“获取”是必要的陈述:它是温度计的一项操作说明。 但是我们不想指示温度计在这里做任何事情; 它已经在做它的工作(测量温度),我们只想知道它的当前读数是多少。 阅读是由我们完成的。 因此,以这种方式编写时,代码更加自然:

Temperature t = thermometer.reading();

我认为这更好地将责任归于责任。 但是请务必考虑是否需要访问器,因为…

对象不是数据结构

用getter和setter编写类的习惯对我们的编码方式有微妙的影响。 它自然化了我们应该进入对象以获取所需数据,对其进行处理,然后使用结果更新对象的想法,而不是让对象自己执行处理。 换句话说,它鼓励我们将对象视为数据包。 我们通过getter提取数据,并通过setter更新它们。 同时,对数据进行操作的代码位于其他位置。

如果我们的编码习惯使我们倾向于将对象视为纯数据结构,那么ORM框架会积极地实施它。 更糟糕的是,如果您使用的是Spring框架–如果您是Java开发人员,那么很有可能–默认情况下,它会将所有bean创建为单例。 (令人困惑的是,Spring bean与JavaBeans无关)。 因此,现在您有了一个由单例对象组成的系统,该系统在无行为的数据结构上运行。 如果您将代码和数据分开听起来像是您所熟悉的编程风格,那您就没错:我们称其为过程编程。

考虑一下这是否是一件好事。 毕竟,Java应该是一种面向对象的编程语言。 OO的一大优点是我们可以编写对象类,其名称和交互作用反映问题域。 它使我们能够编写要解决的问题的代码,而不会模糊基本编程结构和原始数据类型背后的全局。 它帮助我们透过树林看木头。 我们不应该放弃这一点。

该怎么做

尽可能停止编写get and set! 有时这是适当的做法,但一定要停止使用IDE的功能来为您生成getter和setter。 这只是快速执行错误操作的便捷方法。 如果需要在对象上公开某个属性,只需将其命名为该属性,然后还要检查是否确实需要公开该属性。 询问为什么要这样做。 可以将任务委派给对象本身吗? 例如,假设我有一个表示货币金额的类,并且希望对一堆交易进行汇总:

Amount total = new Amount(transactions.stream().map(Transaction::getAmount).mapToDouble(Amount::getValue).sum());

代替getValue访问器,为什么不给Amount类一个add()方法并让它为我求和呢?

Amount total = transactions.stream().map(Transaction::getAmount).reduce(Amount.ZERO, Amount::add);

这带来了好处–也许您对使用双精度表示货币金额的想法很感兴趣。 没错,BigDecimal会更好。 第二个示例使此问题更易于修复,因为内部表示形式得到了更好的封装。 我们只需要在一处更改它。

也许您想要获取对象的数据以测试其是否等于某物。 在这种情况下,请考虑在对象上实现equals()方法,并为您测试是否相等。 如果使用Mockito创建间谍,则无需使用参数捕获器:相反,您可以创建一个等值的对象作为示例,并将其直接传递给verify语句进行比较。

有时您必须创建访问器。 例如,为了将数据持久存储在数据库中,您可能需要访问数据的原始表示形式。 您是否真的必须遵循获取/设置命名约定? 如果您的回答是“这就是用Java完成的方式”,那么我建议您回头阅读JavaBeans规范。 您是否真的在编写JavaBean以按照规范描述的方式使用它? 您是否正在使用期望您的对象遵循约定的框架或库?

必须创建变种器的次数将减少。 函数式编程现在正像一股狂潮一样席卷整个行业,而不变数据的原理是一个很好的原则。 它也应该应用于OO程序。 如果没有必要改变状态,你应该考虑有必要改变状态,所以不用加一个赋值函数方法。 当您编写导致出现新状态的代码时,将尽可能返回新实例来表示新状态。 例如,BigDecimal实例上的算术方法不会更改其自己的值:它们返回表示其结果的新BigDecimal实例。 如今,我们拥有足够的内存和处理能力,可以使这种编程方式变得可行。 而且Spring框架不需要注入依赖方法的setter方法,它也可以通过构造函数参数进行注入。 实际上,这种方法是Spring文档推荐的方法。

某些技术确实要求类遵循JavaBeans约定。 如果您仍在为视图层编写JSP页面,则EL和JSTL期望响应模型对象具有getter方法。 用于将对象与XML进行序列化/反序列化的库可能需要它。 ORM框架可能需要它。 由于这些原因而被迫编写数据结构时,建议您将其隐藏在体系结构边界后面。 当对象泄漏到您的域中时,不要让这些数据结构伪装。

结论

在与使用其他语言工作的程序员交谈时,我经常听到他们批评Java。 他们说“太罗y”或“样板太多”之类的话。 Java当然有其缺陷,但是当我更深入地询问这些批评时,我通常会发现它们针对特定的实践,而不是针对该语言的任何内在实践。 做法不是一成不变的,它们会随着时间的流逝而发展,不良做法可以得到解决。 我认为在Java中随意使用get和set是一种不好的做法,如果放弃,我们会编写更好的代码。

翻译自: https://www.javacodegeeks.com/2018/03/getters-and-setters-considered-harmful.html

空调吸气和排气

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/333900.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Sublime Text for Mac 如何格式化代码

文章目录格式化 HTML/CSS/JS格式化 Java/C/C格式化 HTML/CSS/JS 格式化 HTML/CSS/JS,请安装插件:html-css-js prettify 格式化的快捷键:Shift Cmd H html-css-js prettify 的简介: Usage Tools -> Command Palette (CmdS…

ic启动器我的世界_hmcl启动器下载-我的世界HMCL启动器下载 v3.3.172官方最新版--pc6下载站...

我的世界HMCL启动器是我的世界游戏玩家必备的游戏启动器,是三年来超过使用3亿次的老牌启动器,不需要其他任何设置,操作非常方便,本站提供现在最新版本下载。我的世界HMCL启动器是我的世界游戏玩家必备的游戏启动器,是三…

C++核心编程笔记

C核心编程1 内存分区模型1.1 程序运行前1.2 程序运行后1.3 new操作符2 引用2.1 引用的基本使用2.2 引用注意事项2.3 引用做函数参数2.4 引用做函数返回值2.5 引用的本质2.6 常量引用3 函数提高3.1 函数默认参数3.2 函数占位参数3.3 函数重载3.3.1 函数重载概述3.3.2 函数重载注…

Sublime Text 如何设置组合快捷键

Sublime 有个功能叫再次缩进(Reindent),我就以这个功能为例讲下如何设置快捷键,这个功能的菜单路径是:Edit ➠ Line ➠ Reindent,有人说这个再次缩进可以格式化代码,扯淡,缩进两下也…

朝着理想坚实迈进_坚实原则:开放/封闭原则

朝着理想坚实迈进先前我们讨论了单一责任原则。 关于实体原则首字母缩写, 打开/关闭原则是该行中的第二个原则。 “软件实体(类,模块,功能等)应打开以进行扩展,但应关闭以进行修改” 通过采用该原理&…

协程asyncio_Python 异步模块 asyncio 中的协程与任务

协程(Coroutine)是允许执行被挂起、恢复、以及取消的程序。Python 3 中最初是使用 asyncio.coroutine 装饰器和 yield from 关键字组合来实现协程。单词 yield 在这里并非在生成器(Generator)中所表示的“产出”,而是交…

ie8兼容性视图灰色修复_IE8网页显示不正常 用”兼容性视图”搞定

网页显示不正常,出现图片错位,文字跑远……等等,别急,试试IE8自带的”兼容性视图”功能吧!其实出现网页显示问题,一般不是您的电脑或者浏览器有问题,而是由于各网站开发标准不同,所以在不同的浏…

GAN对抗生成网络原始论文理解笔记

文章目录论文:Generative Adversarial Nets符号意义生成器(Generator)判别器(Discriminator)生成器和判别器的关系GAN的训练流程简述论文中的生成模型和判别模型GAN的数学理论最大似然估计转换为最小化KL散度问题定义PGP_GPG​全局最优论文:Generative A…

php cdi_CDI和lambda的策略模式

php cdi策略设计模式在运行时动态选择一种实现算法,一种策略。 该模式可用于根据情况选择不同的业务算法。 我们可以将不同的算法实现定义为单独的类。 或者,我们利用Java SE 8 lambda和函数,这些lambda和函数在此处用作轻量级策略实现。 C…

Linux 命令之 cp -- 复制文件或目录

文章目录一、命令介绍二、常用选项三、命令示例(一)复制某个目录到某个目录下(二)复制文件(三)复制文件到目标目录下,若存在文件则备份(四)复制某个目录的全部文件到某个…

向上累积频数怎么算_excel数据分析向上累计和向下累计怎么做呢

2016-07-08 00:25赵飞虎 客户经理一、Excel在分析性测试、复核中的运用注册会计师在分析审计风险确定重点审计领域、重要性水平和重大异常经济业务事项时,常常要对被审计单位的会计报表进行分析性测试和复核。在执行具体审计程序时,也常常要对本期数和上…

okta使用_使用Okta的单点登录保护您的Vert.x服务器

okta使用“我喜欢编写身份验证和授权代码。” 〜从来没有Java开发人员。 厌倦了一次又一次地建立相同的登录屏幕? 尝试使用Okta API进行托管身份验证,授权和多因素身份验证。 Vert.x是Spring生态系统中增长最快的元素之一,保护Vert.x服务器可…

Linux 命令之 make -- GNU的工程化编译工具

文章目录一、命令介绍二、常用选项三、命令示例(一)指定命令 make 的工作目录一、命令介绍 make 命令是 GNU 的工程化编译工具,用于编译众多相互关联的源代码文件,还可以编辑内核或模块,以实现工程化的管理&#xff0…

SDL2笔记

SDL2基本操作头文件主函数初始化创建窗口窗口暂停以及事件讲解销毁窗口(释放指针)并退出加载bmp图片新加载图片的方法(使用渲染、纹理)加载其他格式的图片头文件 #include "SDL.h" #include "SDL_image.h"主函数 int main(int argc,char* argv[]) //一定…

操作系统时间片轮换_《操作系统_时间片轮转RR进程调度算法》

转自:https://blog.csdn.net/houchaoqun_xmu/article/details/55540250时间片轮转RR进程调度算法一、概念介绍和案例解析时间片轮转法 - 基本原理:在早期的时间片轮转法中,系统将所有的就绪进程按先来先服务的原则排成一个队列,每次调度时&am…

java ee打印功能_Java EE 8的前5个新功能

java ee打印功能备受期待的Java Enterprise Edition 8版本具有两个令人兴奋的新API(JSON绑定1.0和Java EE Security 1.0),并且对当前API进行了改进(JAX-RS 2.1,Bean Validation 2.0,JSF 2.3,CDI…

SDL2事件笔记

SDL2事件窗口关闭的事件:鼠标事件鼠标点击事件鼠标移动事件键盘事件用SDL_PollEvent(&event)来检测是否有事件,用SDL_Event的实例属性event.type来获取事件。 窗口关闭的事件: SDL_QUIT鼠标事件 鼠标点击事件 鼠标点击事件&#xff1…

什么是复数

我们把形如 zabi(a、b均为实数)的数称为复数。其中,a 称为实部,b 称为虚部,i 称为虚数单位。当 z 的虚部 b=0 时,则 z 为实数;当 z 的虚部 b≠0 时,实部 a=0 …

qnx 设备驱动开发_QNX驱动开发——应用层与resource manger交互 | 学步园

QNX操作系统是一个类Unix实时操作系统,遵从POSIX规范,驱动程序具有良好的可移植性。编写任何驱动程序都会遇到同样的一个问题:应用程序与驱动程序之间是如何进行交互的。其实这个问题很简单,QNX有大量资料说明这一点。当客户端调用…

spring api层打包_Spring项目的按层打包已过时

spring api层打包我认为Spring应用程序不应该以逐层方法构造。 在我看来,按功能打包更有意义。 首先,让我简要描述每种方法。 “按层打包”(在非Java世界中为“按类型折叠”) 该项目结构根据源代码文件所属的体系结构层将其分为…