目前,我正在开发一款带有多种不同模式的Java简单游戏。我已经扩展了一个主Game类,将主要逻辑放在其他类中。尽管如此,游戏的主要职业仍然相当庞大。
在快速浏览了我的代码后,我发现比起游戏逻辑真正需要的其他内容,大部分内容都是getter和setter(60%)。
一些谷歌搜索声称getter和setter是邪恶的,而另一些人则声称它们是良好的OO实践和伟大的程序所必需的。
那我该怎么办呢?应该是哪一个呢?我是否应该为我的私有变量改变我的getter和setter,还是应该继续使用它们?
目前,我正在开发一款带有多种不同模式的Java简单游戏。我已经扩展了一个主Game类,将主要逻辑放在其他类中。尽管如此,游戏的主要职业仍然相当庞大。
在快速浏览了我的代码后,我发现比起游戏逻辑真正需要的其他内容,大部分内容都是getter和setter(60%)。
一些谷歌搜索声称getter和setter是邪恶的,而另一些人则声称它们是良好的OO实践和伟大的程序所必需的。
那我该怎么办呢?应该是哪一个呢?我是否应该为我的私有变量改变我的getter和setter,还是应该继续使用它们?
当前回答
如果Game类暴露了那么多变量,那么它可能遵循god对象反模式。getter和setter并没有什么问题(尽管它们在Java中的冗长可能会有点烦人);在一个设计良好的应用程序中,每个类都有明确分离的功能,你不需要在一个类中包含数十个类。
编辑:如果getter和setter的主要目的是“配置”游戏类(我是这样理解你的评论的),那么你可能不需要getter(一个类不使用get方法就可以访问自己的私有变量),你可能可以将许多setter折叠成“组setter”,设置几个概念上属于一起的变量。
其他回答
他们绝对是邪恶的。
@coobird不幸的是,他们绝对没有“强制封装的概念”,他们所做的只是让你认为你在封装数据,而实际上你是通过一个方法宏伟的属性来暴露数据。getter/setter在公共字段中做的任何事情都做得更好。
首先,如果你想要公共数据,让它成为公共的,摆脱getter和setter方法,以减少客户端必须涉水处理的方法的数量,并使客户端通过eg改变它的值在认知上更简单。
object.field = value;
而不是更强烈的认知
object.setField(value);
客户端现在必须检查getter/setter方法,看看它是否有任何副作用。
其次,如果你真的需要在方法中做一些其他的事情,当它有比简单的获取或设置更多的职责时,为什么要把它称为get/set方法呢? 要么遵循SRP,要么就像Zarkonnen提到的例子一样,给这个方法起个名字,告诉你整个方法是怎么做的。
public void kill(){
isAlive = false;
removeFromWorld(this);
}
而不是
public void setAlive(boolean isAlive){
this.isAlive = isAlive;
if (isAlive)
addToWorld(this);
else
removeFromWorld(this);
}
在哪里setAlive(布尔)方法告诉客户端,作为一个副作用,它将从世界上删除对象?客户端为什么要了解isAlive字段?再加上当对象被重新添加到世界中时会发生什么,它应该被重新初始化吗?客户为什么会关心这些?
恕我直言,这里的寓意是要给方法命名,准确地说出它们做了什么,遵循SRP,去掉getter /setter。 如果在没有getter /setter的情况下出现问题,告诉对象在它们自己的类中做它们自己的脏工作,而不是试图在其他类中与它们一起做事情。
我的咆哮到此结束,对不起;)
这取决于所讨论的编程语言。您的问题是在Java上下文中提出的,在Java上下文中,getter和setter似乎通常被认为是一件好事。
相反,在Python世界中,它们通常被认为是糟糕的风格:它们在代码中添加行,但实际上没有添加功能。当Python程序员需要时,他们可以使用元编程来捕获对象属性的获取和/或设置。
在Java(至少是我十年前学过的Java版本)中,这是不可能的。因此,在Java中最好严格地使用getter和setter,这样如果需要,就可以覆盖对变量的访问。
(这并不意味着Python一定比Java好,只是不同而已。)
如果需要外部访问字段的各个值,请使用getter和/或setter。如果不是,那就不要。永远不要使用公共字段。就这么简单!(好吧,从来没有那么简单,但这是一个很好的经验法则)。
通常情况下,您还会发现您需要提供setter的次数比getter少得多——特别是当您试图使对象不可变时——这是一件好事(但并不总是最好的选择)——但即使不是这样。
关于这个问题,你们已经有了很多很好的答案,所以我只是发表我的意见。getter和setter是非常非常邪恶的。它们本质上让您假装隐藏对象的内部,而大多数情况下您所做的一切都是在多余的代码中进行的,这些代码无法隐藏内部状态。对于一个简单的POJO,没有理由不能用obj.name = "Tom"替换getName()和setName()。
如果方法调用只是替换赋值,那么选择方法调用所得到的只是代码膨胀。不幸的是,该语言在JavaBeans规范中规定了getter和setter的使用,因此Java程序员被迫使用它们,即使这样做毫无意义。
幸运的是,Eclipse(可能还有其他ide)允许您自动生成它们。在一个有趣的项目中,我曾经用XSLT为它们构建了一个代码生成器。但如果有一件事是我想在Java中摆脱的,那就是对getter和setter的过度依赖。
这是滑坡效应。
简单的Transfer对象(或Parameter对象)的唯一目的可能是保存某些字段,并根据需要提供它们的值。然而,即使在这种退化的情况下,也可以认为对象应该是不可变的——在构造函数中配置,并且只公开get…方法。
还有一个类暴露了一些“控制旋钮”;你的汽车收音机的UI可能被理解为公开类似getVolume、setVolume、getChannel和setChannel的东西,但它的真正功能是接收信号和发出声音。但是这些旋钮并没有暴露太多的实现细节;从这些接口特征上,你无法知道收音机是晶体管(主要是软件)还是真空管。
越是开始将对象视为问题域任务的积极参与者,就越会认为是要求它做一些事情,而不是要求它告诉您它的内部状态,或者要求它提供它的数据,以便其他代码可以使用这些值做一些事情。
所以…“邪恶”?不是真的。但每次你倾向于输入一个值并暴露它们都会得到。并设置…方法,问问自己为什么,以及该对象的真正职责是什么。如果你能给自己的唯一答案是,“为我保留这个值”,那么可能有OO之外的东西在这里。