私はこの列挙コードを持っています:
enum Duration { Day, Week, Month };
このEnumの拡張メソッドを追加できますか?
これによると 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で こちら をご覧ください。
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とは異なるタイプであるためにのみ機能します。
もちろん、たとえば、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;
}
すべての答えは素晴らしいですが、彼らは特定の列挙型に拡張メソッドを追加することについて話している。
明示的なキャストの代わりに現在の値の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
object
でも拡張機能を作成できます(ただし、 ベストプラクティスとは見なされません )。拡張メソッドをpublic static
メソッドとして理解してください。メソッドでは、好きなパラメータータイプを使用できます。
public static class DurationExtensions
{
public static int CalculateDistanceBetween(this Duration first, Duration last)
{
//Do something here
}
}
MSDN を参照してください。
public static class Extensions
{
public static string SomeMethod(this Duration enumValue)
{
//Do something here
return enumValue.ToString("D");
}
}
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}";
}
}
この例は無意味なものですが、アイデアは得られます;)