web-dev-qa-db-ja.com

拡張メソッドを列挙に追加する方法

私はこの列挙コードを持っています:

enum Duration { Day, Week, Month };

このEnumの拡張メソッドを追加できますか?

95
user2110292

これによると site

拡張メソッドは、チームの他の人が実際に発見して使用する方法で、既存のクラスのメソッドを記述する方法を提供します。列挙型は他のクラスと同じであるため、次のように列挙型を拡張できることは驚くべきことではありません。

enum Duration { Day, Week, Month };

static class DurationExtensions 
{
  public static DateTime From(this Duration duration, DateTime dateTime) 
  {
    switch (duration) 
    {
      case Day:   return dateTime.AddDays(1);
      case Week:  return dateTime.AddDays(7);
      case Month: return dateTime.AddMonths(1);
      default:    throw new ArgumentOutOfRangeException("duration");
    }
  }
}

列挙型は一般的に最良の選択ではないと思いますが、少なくともこれにより、スイッチ/ if処理の一部を一元化して、何か改善できるまで少し抽象化することができます。値が範囲内であることも忘れずに確認してください。

詳細については、Microsft MSDNで こちら をご覧ください。

90
One Man Crew

Enumのインスタンスではなく、Enum型に拡張メソッドを追加することもできます。

/// <summary> Enum Extension Methods </summary>
/// <typeparam name="T"> type of Enum </typeparam>
public class Enum<T> where T : struct, IConvertible
{
    public static int Count
    {
        get
        {
            if (!typeof(T).IsEnum)
                throw new ArgumentException("T must be an enumerated type");

            return Enum.GetNames(typeof(T)).Length;
        }
    }
}

上記の拡張メソッドを呼び出すには、次を実行します。

var result = Enum<Duration>.Count;

これは真の拡張メソッドではありません。 Enum <>はSystem.Enumとは異なるタイプであるためにのみ機能します。

38
ShawnFeatherly

もちろん、たとえば、DescriptionAttribueの値にenumを使用することもできます。

using System.ComponentModel.DataAnnotations;

public enum Duration 
{ 
    [Description("Eight hours")]
    Day,

    [Description("Five days")]
    Week,

    [Description("Twenty-one days")] 
    Month 
}

今、あなたは次のようなことをできるようにしたい:

Duration duration = Duration.Week;
var description = duration.GetDescription(); // will return "Five days"

拡張メソッドGetDescription()は次のように記述できます。

using System.ComponentModel;
using System.Reflection;

public static string GetDescription(this Enum value)
{
    FieldInfo fieldInfo = value.GetType().GetField(value.ToString());
    if (fieldInfo == null) return null;
    var attribute = (DescriptionAttribute)fieldInfo.GetCustomAttribute(typeof(DescriptionAttribute));
    return attribute.Description;
}
32
Stacked

すべての答えは素晴らしいですが、彼らは特定の列挙型に拡張メソッドを追加することについて話している。

明示的なキャストの代わりに現在の値のintを返すなど、すべての列挙型にメソッドを追加する場合はどうでしょうか。

public static class EnumExtensions
{
    public static int ToInt<T>(this T soure) where T : IConvertible//enum
    {
        if (!typeof(T).IsEnum)
            throw new ArgumentException("T must be an enumerated type");

        return (int) (IConvertible) soure;
    }

    //ShawnFeatherly funtion (above answer) but as extention method
    public static int Count<T>(this T soure) where T : IConvertible//enum
    {
        if (!typeof(T).IsEnum)
            throw new ArgumentException("T must be an enumerated type");

        return Enum.GetNames(typeof(T)).Length;
    }
}

IConvertibleの背後にあるトリックは、その継承階層を参照してください MDSN

ShawnFeatherlyの回答に感謝

24

objectでも拡張機能を作成できます(ただし、 ベストプラクティスとは見なされません )。拡張メソッドをpublic staticメソッドとして理解してください。メソッドでは、好きなパラメータータイプを使用できます。

public static class DurationExtensions
{
  public static int CalculateDistanceBetween(this Duration first, Duration last)
  {
    //Do something here
  }
}
7
Tim Schmelter

MSDN を参照してください。

public static class Extensions
{
  public static string SomeMethod(this Duration enumValue)
  {
    //Do something here
    return enumValue.ToString("D"); 
  }
}
5
LukeHennerley

c#の列挙型拡張機能を作成しました https://github.com/simonmau/enum_ext

これはタイプセーフの実装にすぎませんが、うまく機能するので、共有するパッケージを作成しました-楽しんでください

public sealed class Weekday : TypeSafeNameEnum<Weekday, int>
{
    public static readonly Weekday Monday = new Weekday(1, "--Monday--");
    public static readonly Weekday Tuesday = new Weekday(2, "--Tuesday--");
    public static readonly Weekday Wednesday = new Weekday(3, "--Wednesday--");
    ....

    private Weekday(int id, string name) : base(id, name)
    {
    }

    public string AppendName(string input)
    {
        return $"{Name} {input}";
    }
}

この例は無意味なものですが、アイデアは得られます;)

1
simonmau