namespace Immutable { using System; public interface Option { U Match_(Func some, Func none); } public static class Option { public static Option Some(T value) => new Types.Some(value); public static Option None() => new Types.None(); private static class Types { public class Some : Option { public readonly T value; public Some(T value) { this.value = value; } public U Match_(Func Some, Func None) => Some(value); } public class None : Option { public None() { } public U Match_(Func Some, Func None) => None(); } } } public static class OptionExtensionMethods { public static Option Some(this T value) => Option.Some(value); public static U Match(this Option o, Func some, Func none) => o.Match_(some, none); public static U Match(this Option o, Func some, U none) => o.Match_(some, () => none); } }