我正在研究模式和反模式。我对模式有一个清晰的概念,但我没有反模式。来自网络和维基百科的定义让我很困惑。

有人能用简单的话给我解释一下什么是反模式吗?目的是什么?他们做什么?这是好事还是坏事?


当前回答

任何对给定的软件开发环境弊大于利的设计模式都被认为是反模式。

有些反模式是明显的,有些则不是。例如,单例,尽管许多人认为这是一个很好的旧设计模式,但也有其他人不这么认为。

你可以检查问题单身有什么不好?为了更好地理解对它的不同意见。

其他回答

今天,软件工程研究人员和实践者经常互换使用术语“反模式”和“气味”。然而,它们在概念上并不相同。维基百科上关于反模式的条目指出,反模式与坏实践或坏想法至少有两个不同之处。反模式是

一种常用的过程、结构或行为模式 最初看起来是对一种疾病的适当而有效的反应 问题,通常有更多的坏后果比有益的结果。”

它清楚地表明,选择反模式是因为相信它是当前问题的一个很好的解决方案(作为模式);然而,它带来的负债多于收益。另一方面,气味只是一种糟糕的实践,它会对软件系统的质量产生负面影响。例如,Singleton是一种反模式,God类(或不足模块化)是一种设计气味。

如果您确实希望学习反模式,请阅读《反模式》(ISBN-13: 978-0471197133)这本书。

在这篇文章中,他们定义“反模式是一种文学形式,它描述了一个问题的常见解决方案,该解决方案会产生明显的负面后果。”

因此,如果这是一种糟糕的编程实践,但并不常见——出现的频率非常有限,那么它就不符合AntiPattern定义中的“模式”部分。

反模式是软件开发中被认为是糟糕编程实践的某些模式。

设计模式是解决常见问题的常用方法,已经形式化,通常被认为是一种很好的开发实践,与之相反,反模式是不可取的。

例如,在面向对象编程中,思想是将软件分割成称为对象的小块。面向对象编程中的反模式是一个上帝对象,它执行许多功能,最好将这些功能分离到不同的对象中。

例如:

class GodObject {
    function PerformInitialization() {}
    function ReadFromFile() {}
    function WriteToFile() {}
    function DisplayToScreen() {}
    function PerformCalculation() {}
    function ValidateInput() {}
    // and so on... //
}

上面的例子有一个对象可以做所有事情。在面向对象编程中,最好为不同的对象定义明确的职责,以减少代码的耦合,最终更易于维护:

class FileInputOutput {
    function ReadFromFile() {}
    function WriteToFile() {}
}

class UserInputOutput {
    function DisplayToScreen() {}
    function ValidateInput() {}
}

class Logic {
    function PerformInitialization() {}
    function PerformCalculation() {}
}

总的来说,有很多使用常用模式(设计模式)开发软件的好方法,但是也有一些开发和实现软件的方法可能会导致问题。被认为是糟糕的软件开发实践的模式是反模式。

这是制造混乱的常见方法。例如,god/厨房墨水类(做所有事情)。

每当我听到反模式,我就会想起另一个术语,即设计气味。

“设计气味是指设计中违反基本设计原则并对设计质量产生负面影响的某些结构。”(摘自《软件设计的重构:管理技术债务》)

根据违反设计原则,有许多设计气味分类:

抽象的气味

缺少抽象:当使用数据块或编码字符串而不是创建类或接口时,就会出现这种情况。

命令式抽象:当操作被转换为类时,就会产生这种气味。

不完全抽象:当抽象不完全支持互补或相互关联的方法时,就会出现这种气味。

多面抽象:当一个抽象被分配了不止一个职责时,就会产生这种气味。

不必要的抽象:当软件设计中引入了实际上不需要的抽象(因此可以避免)时,就会产生这种气味。

未被利用的抽象:当抽象未被使用(要么没有直接使用,要么无法到达)时,就会产生这种气味。

重复抽象:当两个或多个抽象具有相同的名称或相同的实现或两者都具有时,就会出现这种气味。

封装的气味

封装缺陷:当抽象的一个或多个成员声明的可访问性比实际需要的更允许时,就会发生这种气味。

泄漏封装:当抽象通过其公共接口“暴露”或“泄漏”实现细节时,就会产生这种气味。

缺少封装:当实现变量没有封装在抽象或层次结构中时,就会发生这种情况。

未利用封装:当客户端代码使用显式类型检查(使用链式if-else或switch语句检查对象类型)而不是利用已经封装在层次结构中的类型变化时,就会产生这种气味。

模块化的气味

破碎的模块化:当理想情况下应该本地化到单个抽象中的数据和/或方法被分离并分散到多个抽象中时,就会出现这种气味。

模块化不足:当抽象尚未完全分解时就会产生这种气味,而进一步的分解可能会减少其大小、实现复杂性或两者兼有。

周期依赖的模块化:当两个或多个抽象直接或间接地相互依赖(在抽象之间创建紧密耦合)时,就会出现这种气味。

类似hub的模块化:当一个抽象与大量其他抽象具有依赖关系(包括传入和传出)时,就会出现这种气味。

层次结构的气味

缺少层次结构:当代码段使用条件逻辑(通常与“标记类型”结合在一起)显式管理行为中的变化时,就会出现这种气味,而在这种情况下,可以创建层次结构并用于封装这些变化。

不必要的层次结构:当整个继承层次结构都是不必要的时,就会出现这种气味,这表明在特定的设计上下文中不必要地应用了继承。

未分解层次结构:当层次结构中的类型之间存在不必要的重复时,就会产生这种气味。

广泛的层次结构:当继承层次结构“太”宽,表明中间类型可能缺失时,就会出现这种气味。

投机层次结构:当一个层次结构中的一个或多个类型被投机地提供(即,基于想象的需求而不是真实的需求)时,这种气味就会出现。

深层层次结构:当继承层次结构“过度”深时,就会产生这种气味。

叛逆层次:当子类型拒绝其超类型提供的方法时,就会出现这种味道。

破碎的层次结构:当超类型及其子类型在概念上不共享“IS- a”关系,导致可替代性破碎时,就会出现这种气味。

多路径层次结构:当子类型直接或间接地继承超类型,导致层次结构中出现不必要的继承路径时,就会产生这种气味。

循环层次结构:当层次结构中的超类型依赖于它的任何子类型时,就会出现这种气味。


上面的定义和分类在“软件设计气味的重构:管理技术债务”中有描述。更多相关的资源可以在这里找到。