除了使用访问者模式之外,有没有办法在C#中使用变体?

前端之家收集整理的这篇文章主要介绍了除了使用访问者模式之外,有没有办法在C#中使用变体?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
在C#中没有直接支持 variant types(也称为标记的联合,有区别的联合).然而,可以使用 visitor pattern,通过双重调度实现歧视,并保证在编译时解决所有情况.然而,实施起来很繁琐.我想知道是否有更容易获得的方法:某种具有歧视机制的变体可以保证在C#的编译时解决所有联合的情况?
  1. // This is a variant type. At each single time it can only hold one case (a value)
  2. // from a predefined set of cases. All classes that implement this interface
  3. // consitute the set of the valid cases of the variant. So at each time a variant can
  4. // be an instance of one of the classes that implement this interface. In order to
  5. // add a new case to the variant there must be another class that implements
  6. // this interface.
  7. public interface ISomeAnimal
  8. {
  9. // This method introduces the currently held case to whoever uses/processes
  10. // the variant. By processing we mean that the case is turned into a resulting
  11. // value represented by the generic type TResult.
  12. TResult GetProcessed<TResult>(ISomeAnimalProcessor<TResult> processor);
  13. }
  14.  
  15. // This is the awkward part,the visitor that is required every time we want to
  16. // to process the variant. For each possible case this processor has a corresponding
  17. // method that turns that case to a resulting value.
  18. public interface ISomeAnimalProcessor<TResult>
  19. {
  20. TResult ProcessCat(Cat cat);
  21. TResult ProcessFish(Fish fish);
  22. }
  23.  
  24. // A case that represents a cat from the ISomeAnimal variant.
  25. public class Cat : ISomeAnimal
  26. {
  27. public CatsHead Head { get; set; }
  28. public CatsBody Body { get; set; }
  29. public CatsTail Tail { get; set; }
  30. public IEnumerable<CatsLeg> Legs { get; set; }
  31. public TResult GetProcessed<TResult>(ISomeAnimalProcessor<TResult> processor)
  32. {
  33. // a processor has a method for each case of a variant,for this
  34. // particular case (being a cat) we always pick the ProcessCat method
  35. return processor.ProcessCat(this);
  36. }
  37. }
  38.  
  39. // A case that represents a fish from the ISomeAnimal variant.
  40. public class Fish : ISomeAnimal
  41. {
  42. public FishHead Head { get; set; }
  43. public FishBody Body { get; set; }
  44. public FishTail Tail { get; set; }
  45. public TResult GetProcessed<TResult>(ISomeAnimalProcessor<TResult> processor)
  46. {
  47. // a processor has a method for each case of a variant,for this
  48. // particular case (being a fish) we always pick the ProcessCat method
  49. return processor.ProcessFish(this);
  50. }
  51. }
  52.  
  53. public static class AnimalPainter
  54. {
  55. // Now,in order to process a variant,in this case we want to
  56. // paint a picture of whatever animal it prepresents,we have to
  57. // create a new implementation of ISomeAnimalProcessor interface
  58. // and put the painting logic in it.
  59. public static void AddAnimalToPicture(Picture picture,ISomeAnimal animal)
  60. {
  61. var animalToPictureAdder = new AnimalToPictureAdder(picture);
  62. animal.GetProcessed(animalToPictureAdder);
  63. }
  64.  
  65. // Making a new visitor every time you need to process a variant:
  66. // 1. Requires a lot of typing.
  67. // 2. Bloats the type system.
  68. // 3. Makes the code harder to maintain.
  69. // 4. Makes the code less readable.
  70. private class AnimalToPictureAdder : ISomeAnimalProcessor<Nothing>
  71. {
  72. private Picture picture;
  73.  
  74. public AnimalToPictureAdder(Picture picture)
  75. {
  76. this.picture = picture;
  77. }
  78.  
  79. public Nothing ProcessCat(Cat cat)
  80. {
  81. this.picture.AddBackground(new SomeHouse());
  82. this.picture.Add(cat.Body);
  83. this.picture.Add(cat.Head);
  84. this.picture.Add(cat.Tail);
  85. this.picture.AddAll(cat.Legs);
  86. return Nothing.AtAll;
  87. }
  88.  
  89. public Nothing ProcessFish(Fish fish)
  90. {
  91. this.picture.AddBackground(new SomeUnderwater());
  92. this.picture.Add(fish.Body);
  93. this.picture.Add(fish.Tail);
  94. this.picture.Add(fish.Head);
  95. return Nothing.AtAll;
  96. }
  97. }
  98.  
  99. }

解决方法

你在寻找 Boost Variants的东西吗?如果是这样,我认为不可能直接移植,因为C模板语言和C#泛型有些不同.此外,boost :: variant使用访问者模式.无论如何,如果你愿意,你可以写类似的东西.例如(请注意,此代码仅是概念证明),您可以为访问者和变体定义两种通用类型:
  1. public interface VariantVisitor<T,U>
  2. {
  3. void Visit(T item);
  4. void Visit(U item);
  5. }
  6.  
  7. public class Variant<T,U>
  8. {
  9. public T Item1 { get; private set; }
  10. private bool _item1Set;
  11. public U Item2 { get; private set; }
  12. private bool _item2Set;
  13.  
  14. public Variant()
  15. {
  16. }
  17.  
  18. public void Set(T item)
  19. {
  20. this.Item1 = item;
  21. _item1Set = true;
  22. _item2Set = false;
  23. }
  24.  
  25. public void Set(U item)
  26. {
  27. this.Item2 = item;
  28. _item1Set = false;
  29. _item2Set = true;
  30. }
  31.  
  32. public void ApplyVisitor(VariantVisitor<T,U> visitor)
  33. {
  34. if (_item1Set)
  35. {
  36. visitor.Visit(this.Item1);
  37. }
  38. else if (_item2Set)
  39. {
  40. visitor.Visit(this.Item2);
  41. }
  42. else
  43. {
  44. throw new InvalidOperationException("Variant not set");
  45. }
  46. }
  47. }

你可以使用这样的类型:

  1. private static object _result;
  2.  
  3. internal class TimesTwoVisitor : VariantVisitor<int,string>
  4. {
  5. public void Visit(int item)
  6. {
  7. _result = item * 2;
  8. }
  9.  
  10. public void Visit(string item)
  11. {
  12. _result = item + item;
  13. }
  14. }
  15.  
  16. [Test]
  17. public void TestVisitVariant()
  18. {
  19. var visitor = new TimesTwoVisitor();
  20. var v = new Variant<int,string>();
  21.  
  22. v.Set(10);
  23. v.ApplyVisitor(visitor);
  24. Assert.AreEqual(20,_result);
  25.  
  26. v.Set("test");
  27. v.ApplyVisitor(visitor);
  28. Assert.AreEqual("testtest",_result);
  29.  
  30. var v2 = new Variant<double,DateTime>();
  31. v2.Set(10.5);
  32. //v2.ApplyVisitor(visitor);
  33. // Argument 1: cannot convert from 'TestCS.TestVariant.TimesTwoVisitor' to 'TestCS.TestVariant.VariantVisitor<double,System.DateTime>'
  34. }

这样,编译器可以验证您是否将正确的访问者传递给正确的变体,并且VariantVisitor接口强制您为变体的所有类型实现Visit方法.显然,您还可以使用两个以上的参数定义变体:

  1. public interface VariantVisitor<T,U,V>
  2. ...
  3. public interface VariantVisitor<T,V,W>
  4. ...
  5.  
  6. public class Variant<T,V>
  7. ...
  8. public class Variant<T,W>
  9. ...

但我个人不喜欢这种方法,我宁愿将Visit方法转换为lambdas,并在需要时将它们作为参数传递,如上面的评论中所指出的那样.例如,您可以编写某种穷人的模式匹配,将此方法添加到类Variant< T,U>:

  1. public R Match<R>(Func<T,R> f1,Func<U,R> f2)
  2. {
  3. if (_item1Set)
  4. {
  5. return f1(this.Item1);
  6. }
  7. else if (_item2Set)
  8. {
  9. return f2(this.Item2);
  10. }
  11. else
  12. {
  13. throw new InvalidOperationException("Variant not set");
  14. }
  15. }

并像这样使用它:

  1. [Test]
  2. public void TestMatch()
  3. {
  4. var v = new Variant<int,string>();
  5.  
  6. v.Set(10);
  7. var r1 = v.Match(
  8. i => i * 2,s => s.Length);
  9. Assert.AreEqual(20,r1);
  10.  
  11. v.Set("test");
  12. var r2 = v.Match(
  13. i => i.ToString(),s => s + s);
  14. Assert.AreEqual("testtest",r2);
  15. }

但请注意,真正的模式匹配具有更多功能:警卫,详尽检查,脆弱的模式匹配检查等.

猜你在找的C#相关文章