目前,我正在开发一款带有多种不同模式的Java简单游戏。我已经扩展了一个主Game类,将主要逻辑放在其他类中。尽管如此,游戏的主要职业仍然相当庞大。

在快速浏览了我的代码后,我发现比起游戏逻辑真正需要的其他内容,大部分内容都是getter和setter(60%)。

一些谷歌搜索声称getter和setter是邪恶的,而另一些人则声称它们是良好的OO实践和伟大的程序所必需的。

那我该怎么办呢?应该是哪一个呢?我是否应该为我的私有变量改变我的getter和setter,还是应该继续使用它们?


当前回答

您可能希望用值类替换一些类。这将允许您删除getter,并避免在更改下面的内容时出现问题。

其他回答

非常邪恶:公共领域。 有点邪恶:在不需要的地方使用getter和setter。 好:getter和setter只在真正需要的地方使用——使类型暴露“更大”的行为,恰好使用它的状态,而不是仅仅将类型视为供其他类型操作的状态存储库。

不过,这真的取决于情况-有时您真的只是想要一个愚蠢的数据对象。

我已经用java编程了几个月了,我知道我们应该只在应用程序需要的时候才使用getter和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的情况下出现问题,告诉对象在它们自己的类中做它们自己的脏工作,而不是试图在其他类中与它们一起做事情。

我的咆哮到此结束,对不起;)

如果Game类暴露了那么多变量,那么它可能遵循god对象反模式。getter和setter并没有什么问题(尽管它们在Java中的冗长可能会有点烦人);在一个设计良好的应用程序中,每个类都有明确分离的功能,你不需要在一个类中包含数十个类。

编辑:如果getter和setter的主要目的是“配置”游戏类(我是这样理解你的评论的),那么你可能不需要getter(一个类不使用get方法就可以访问自己的私有变量),你可能可以将许多setter折叠成“组setter”,设置几个概念上属于一起的变量。

我并不真的认为他们是邪恶的。但我很想生活在一个除非真的需要,我才会用到它们的世界里。

上面我读到的一个例子是对代码进行未来验证。例如:

public void setValue(int value)
{
    this.value = value;
}

然后,需求发生变化,您需要跟踪该值设置了多少次。

So:

public void setValue(int value)
{
    this.value = value;
    count++;
}

真漂亮。我明白了。但是,在Ruby中,下面的代码不能达到同样的目的吗?

someobject.my_value = 100

稍后,您需要跟踪my_value被设置的次数。那么,你能不重写setter then和only then吗?

def my_value=(value)
    @my_value = value
    @count++
end

我很喜欢漂亮的代码,但我不得不承认,在我们拥有的Java类的大山中,看到成千上万行只是基本的getter/setter的代码是丑陋和讨厌的。

当我全职用c#开发时,我们一直使用公共属性,只在需要时才使用自定义getter /setter。效果很好,没有损坏任何东西。