按照设计,java中的枚举常量是单例,为了并发使用,我通常会创建无状态枚举实例,并使用方法参数根据需要注入数据。
示例:
目前我正在创建一个具有Operations的REST服务(使用策略模式的变体实现为枚举)。
public enum Operation {
DO_THIS() {
public Result doSomething(Object theData) {
}
} ,
// Other Operations go here
;
public abstract Result doSomething(Object theData);
}
现在我想收集有关操作被调用频率以及成功频率的数据等。
我可以在使用枚举实例时从外部保存状态,但似乎状态应该保存在Operation中,因为操作应该包含它自己的状态。
现在我的一般问题是:
有状态枚举实例(除了并发问题)是一个糟糕的设计吗?
答案 0 :(得分:18)
我认为它违反了最小惊讶原则。
人们期望枚举的常见用法是最初设计的 - 作为常量或标记,而不是具有状态的通用类。
答案 1 :(得分:11)
每次创建可变枚举时,小猫都会死亡。保存小猫!
答案 2 :(得分:7)
是。而'是'我的意思是'永远'。
如果要整理所调用操作数的统计信息,请实现一些可观察性。
答案 3 :(得分:4)
任何形式的可变静态都是罪。 (好吧,你可能会使用非漏洞缓存,一些懒惰的初始化和日志记录形式。)
答案 4 :(得分:4)
有状态的枚举是矛盾的,甚至是反模式!
http://en.wikipedia.org/wiki/Enumeration
枚举是项目的集合,是该集合中所有项目的完整有序列表。该术语通常用于数学和理论计算机科学,以指代一组中所有元素的列表。在统计中,使用术语分类变量而不是枚举。枚举的精确要求(例如,集合是否必须是有限的,或者列表是否允许包含重复)取决于数学的分支和工作的上下文。
枚举具有有限数量的值,它们应该是常量,它们是。
然而,它们是“一流”Java对象这一事实完全违背了枚举的意图或精神。
如果需要任何类型的状态,枚举(如前所述)应该在Aspect或违规枚举中保持状态,应该在非常实际的情况下保持对委托类保持状态的引用。理解“关注点分离”将有所帮助。
答案 5 :(得分:1)
这对于枚举来说似乎很糟糕 - 为什么不只是为每个操作使用带有新子类的基本抽象类?
答案 6 :(得分:1)
我完全同意mparaz它违反了最小惊讶原则。人们期望枚举成为常数。
您几乎可以肯定地通过以下方式处理日志记录:
DO_THIS() {
public Result doSomething(Object theData) {
MyUtilClass.doSomething(Object theData);
}
}
并将您的日志记录放在另一个类中。
然而,如果你不能解决这个问题,最小惊讶原则就是一个指导原则;你可以违反它提供给你的班级用户足够警告正在发生的事情。确保Enum声明包含一个BIG通知,说明它是可变的,并准确描述了可变性是什么。恩欧姆仍然应该工作;它正在对单个实例进行参考比较以测试枚举值。
答案 7 :(得分:0)
有一种情况可能是合理的。 枚举可以实现一个接口,通常考虑到特定的用例,可以让你创建 在运行时/公开"一些其他类型的枚举类"以动态的方式,将它命名为。
这意味着enum" singleton"实例可以被强制实现一些可变的预期方法签名(作为setter),当然,你仍然可以使用空代码或NotSupportedException来隐藏。
幸运的是,界面中的最终方法不允许任何改变状态的可能性。那本来是唯一可以理解的"我可以想出来。