在C#中表示参数化枚举的最佳方法?

Mic*_*kin 1 c# enums ocaml haxe

是否有任何好的解决方案来表示参数化的枚举C# 3.0?我正在寻找像OCamlHaxe这样的东西.我只能想到具有简单枚举字段的类层次结构,以便于现在轻松切换,也许有更好的想法?

请参阅下面的其中一个回复中的Ocaml示例,以下是Haxe代码:

enum Tree {
   Node(left: Tree, right: Tree);
   Leaf(val: Int);
}
Run Code Online (Sandbox Code Playgroud)

Ori*_*rds 7

不熟悉OCaml或Haxe,并且不够聪明地理解其他解释,我去查找Haxe枚举文档 - 底部的'枚举类型参数'位似乎是相关部分.

我的理解基于以下内容:

"普通"枚举基本上是一个值,它仅限于您在枚举定义中定义的内容.C#示例:

enum Color{ Red, Green, Yellow, Blue };
Color c = Color.Red;
Run Code Online (Sandbox Code Playgroud)

c可以是Red,Green,Yellow,或Blue,但没有别的.

在Haxe中,您可以在枚举中添加复杂类型,从其页面中获取Contrived示例:

enum Cell<T>{ 
  empty; 
  cons( item : T, next : Cell<T> )
}

Cell<int> c = <I don't know>;
Run Code Online (Sandbox Code Playgroud)

似乎意味着它c被限制为文字值empty(就像我们过时的C#枚举),或者它也可以是一个复杂的类型cons(item, next),其中itema Tnexta是Cell<T>.

从来没有使用它看起来它可能会生成一些匿名类型(比如C#编译器在你做的时候会这样做new { Name='Joe'}.
每当你访问'枚举值时,你必须声明item,next当你这样做时,它看起来像它们绑定到临时局部变量.

Haxe示例 - 您可以看到'next'被用作临时局部变量来从匿名cons结构中提取数据:

switch( c ) {
  case empty : 0;
  case cons(item,next): 1 + cell_length(next);
}
Run Code Online (Sandbox Code Playgroud)

说实话,当我"点击"它似乎正在做的事情时,这让我大吃一惊.它似乎非常强大,我可以看到你为什么要在C#中寻找类似的功能.

C#枚举与最初复制它们的C/++枚举几乎相同.它基本上是一种很好的说法,#define Red 1因此当您传递Color对象时,编译器可以使用整数而不是字符串进行比较和存储.

我在C#中做这件事就是使用泛型和接口.像这样的东西:

public interface ICell<T> {
   T Item{ get; set; }
   ICell<T>{ get; set; }
}

class Cons<T> : ICell<T> {
  public T Item{ get; set; } /* C#3 auto-backed property */
  public Cell<T> Next{ get; set; }
}

class EmptyCell<T> : ICell<T>{
  public T Item{ get{ return default(T); set{ /* do nothing */ }; }
  public ICell<T> Next{ get{ return null }; set{ /* do nothing */; }
}
Run Code Online (Sandbox Code Playgroud)

然后你可以有一个List<ICell<T>>包含项目和下一个单元格,并且你可以EmptyCell在最后插入(或者只是将Next引用显式设置为null).优点是因为不EmptyCell包含任何成员变量,所以不需要任何存储空间(如emptyHaxe),而Cons单元格则需要.
编译器也可以内联/优化方法,EmptyCell因为它们什么也不做,因此可能会有一个速度增加,而不仅仅是将Cons其成员数据设置为null.

我真的不知道.我欢迎任何其他可能的解决方案,因为我不是特别自豪我的一个:-)