たとえば 'Gender
'(Male =0 , Female =1
)の列挙型があり、独自のGender列挙型(Male =0 , Female =1, Unknown =2
)を持つサービスから別の列挙型があります
私の質問は、列挙型から私のものに変換するために、すばやく簡単に何かを書くにはどうすればよいですか?
Nateが提案する2つの変換メソッドを使用する場合、拡張メソッドを使用すると非常にうまく機能します。
public static class TheirGenderExtensions
{
public static MyGender ToMyGender(this TheirGender value)
{
// insert switch statement here
}
}
public static class MyGenderExtensions
{
public static TheirGender ToTheirGender(this MyGender value)
{
// insert switch statement here
}
}
明らかにしたくない場合は、別個のクラスを使用する必要はありません。私の好みは、拡張メソッドを適用するクラス/構造/列挙によってグループ化することです。
Enum1 value = ...
が与えられた場合、名前を意味する場合:
Enum2 value2 = (Enum2) Enum.Parse(typeof(Enum2), value.ToString());
数値を意味する場合、通常は次のようにキャストできます。
Enum2 value2 = (Enum2)value;
(キャストでは、Enum.IsDefined
を使用して有効な値を確認することもできます)
1つをintにキャストしてから、他の列挙型にキャストします(値に基づいてマッピングを実行することを考慮してください)。
Gender2 gender2 = (Gender2)((int)gender1);
徹底的にするために、通常はEnum 1を受け取りEnum 2を返す関数と、Enum 2を受け取りEnum 1を返す関数のペアを作成します。各関数は入力を出力にマッピングするcaseステートメントで構成され、デフォルトのcaseは例外をスローします予期しない値について文句を言うメッセージ。
この特定のケースでは、MaleとFemaleの整数値が同じであるという事実を利用できますが、いずれかのenumが将来変更された場合、それはハック的で破損しやすいので、私はそれを避けます。
私たちが持っている場合:
enum Gender
{
M = 0,
F = 1,
U = 2
}
そして
enum Gender2
{
Male = 0,
Female = 1,
Unknown = 2
}
安全にできる
var gender = Gender.M;
var gender2 = (Gender2)(int)gender;
あるいは
var enumOfGender2Type = (Gender2)0;
「=」記号の右側の列挙型が左側の列挙型よりも多くの値を持つ場合をカバーしたい場合-他の人が示唆したように、それをカバーする独自のメソッド/辞書を作成する必要があります。
このような単純な汎用拡張メソッドを書くことができます
public static T ConvertTo<T>(this object value)
where T : struct,IConvertible
{
var sourceType = value.GetType();
if (!sourceType.IsEnum)
throw new ArgumentException("Source type is not enum");
if (!typeof(T).IsEnum)
throw new ArgumentException("Destination type is not enum");
return (T)Enum.Parse(typeof(T), value.ToString());
}
次のような単純な関数を作成できます。
public static MyGender ConvertTo(TheirGender theirGender)
{
switch(theirGender)
{
case TheirGender.Male:
break;//return male
case TheirGender.Female:
break;//return female
case TheirGender.Unknown:
break;//return whatever
}
}
誰かが興味がある場合の拡張メソッドのバージョンはこちら
public static TEnum ConvertEnum<TEnum >(this Enum source)
{
return (TEnum)Enum.Parse(typeof(TEnum), source.ToString(), true);
}
// Usage
NewEnumType newEnum = oldEnumVar.ConvertEnum<NewEnumType>();
しばらく前に、いくつかの異なるEnum
sで機能するセット拡張メソッドを作成しました。特に、実行しようとしているものに対して機能し、Enum
でFlagsAttribute
sを処理します。また、さまざまな基本型でEnum
sを処理します。
public static tEnum SetFlags<tEnum>(this Enum e, tEnum flags, bool set, bool typeCheck = true) where tEnum : IComparable
{
if (typeCheck)
{
if (e.GetType() != flags.GetType())
throw new ArgumentException("Argument is not the same type as this instance.", "flags");
}
var flagsUnderlyingType = Enum.GetUnderlyingType(typeof(tEnum));
var firstNum = Convert.ToUInt32(e);
var secondNum = Convert.ToUInt32(flags);
if (set)
firstNum |= secondNum;
else
firstNum &= ~secondNum;
var newValue = (tEnum)Convert.ChangeType(firstNum, flagsUnderlyingType);
if (!typeCheck)
{
var values = Enum.GetValues(typeof(tEnum));
var lastValue = (tEnum)values.GetValue(values.Length - 1);
if (newValue.CompareTo(lastValue) > 0)
return lastValue;
}
return newValue;
}
そこから、他のより具体的な拡張メソッドを追加できます。
public static tEnum AddFlags<tEnum>(this Enum e, tEnum flags) where tEnum : IComparable
{
SetFlags(e, flags, true);
}
public static tEnum RemoveFlags<tEnum>(this Enum e, tEnum flags) where tEnum : IComparable
{
SetFlags(e, flags, false);
}
これはあなたがやろうとしているようにEnum
sのタイプを変更します。
public static tEnum ChangeType<tEnum>(this Enum e) where tEnum : IComparable
{
return SetFlags(e, default(tEnum), true, false);
}
ただし、フラグを持たないものであっても、このメソッドを使用してEnum
と他のEnum
の間で変換できることに注意してください。例えば:
public enum Turtle
{
None = 0,
Pink,
Green,
Blue,
Black,
Yellow
}
[Flags]
public enum WriteAccess : short
{
None = 0,
Read = 1,
Write = 2,
ReadWrite = 3
}
static void Main(string[] args)
{
WriteAccess access = WriteAccess.ReadWrite;
Turtle turtle = access.ChangeType<Turtle>();
}
変数turtle
の値はTurtle.Blue
になります。
ただし、このメソッドを使用すると、未定義のEnum
値から安全になります。例えば:
static void Main(string[] args)
{
Turtle turtle = Turtle.Yellow;
WriteAccess access = turtle.ChangeType<WriteAccess>();
}
この場合、access
WriteAccess
の最大値は3なので、Enum
はWriteAccess.ReadWrite
に設定されます。
Enum
sとFlagsAttribute
を混在させることの別の副作用は、変換プロセスで値が1対1で一致しないことです。
public enum Letters
{
None = 0,
A,
B,
C,
D,
E,
F,
G,
H
}
[Flags]
public enum Flavors
{
None = 0,
Cherry = 1,
Grape = 2,
Orange = 4,
Peach = 8
}
static void Main(string[] args)
{
Flavors flavors = Flavors.Peach;
Letters letters = flavors.ChangeType<Letters>();
}
この場合、Letters.H
のバッキング値は8であるため、letters
の値はLetters.D
ではなくFlavors.Peach
になります。また、Flavors.Cherry | Flavors.Grape
からLetters
への変換は、 Letters.C
、これは直感的ではないように見える場合があります。
Justin's 上記の答えに基づいて、私はこれを思いつきました:
/// <summary>
/// Converts Enum Value to different Enum Value (by Value Name) See https://stackoverflow.com/a/31993512/6500501.
/// </summary>
/// <typeparam name="TEnum">The type of the enum to convert to.</typeparam>
/// <param name="source">The source enum to convert from.</param>
/// <returns></returns>
/// <exception cref="InvalidOperationException"></exception>
public static TEnum ConvertTo<TEnum>(this Enum source)
{
try
{
return (TEnum) Enum.Parse(typeof(TEnum), source.ToString(), ignoreCase: true);
}
catch (ArgumentException aex)
{
throw new InvalidOperationException
(
$"Could not convert {source.GetType().ToString()} [{source.ToString()}] to {typeof(TEnum).ToString()}", aex
);
}
}
public static TEnum ConvertByName<TEnum>(this Enum source, bool ignoreCase = false) where TEnum : struct
{
// if limited by lack of generic enum constraint
if (!typeof(TEnum).IsEnum)
{
throw new InvalidOperationException("enumeration type required.");
}
TEnum result;
if (!Enum.TryParse(source.ToString(), ignoreCase, out result))
{
throw new Exception("conversion failure.");
}
return result;
}
私はそれが古い質問であり、多くの答えがあることを知っていますが、受け入れられた答えのようにswitchステートメントを使用するのはやや面倒なので、ここに私の2セントがあります:
私の個人的なお気に入りの方法は、キーがソース列挙型で値がターゲット列挙型である辞書を使用することです-そのため、質問で提示された場合、私のコードは次のようになります:
var genderTranslator = new Dictionary<TheirGender, MyGender>();
genderTranslator.Add(TheirGender.Male, MyGender.Male);
genderTranslator.Add(TheirGender.Female, MyGender.Female);
genderTranslator.Add(TheirGender.Unknown, MyGender.Unknown);
// translate their to mine
var myValue = genderTranslator[TheirValue];
// translate mine to their
var TheirValue = genderTranslator .FirstOrDefault(x => x.Value == myValue).Key;;
もちろん、これは静的クラスにラップして、拡張メソッドとして使用できます。
public static class EnumTranslator
{
private static Dictionary<TheirGender, MyGender> GenderTranslator = InitializeGenderTranslator();
private static Dictionary<TheirGender, MyGender> InitializeGenderTranslator()
{
var translator = new Dictionary<TheirGender, MyGender>();
translator.Add(TheirGender.Male, MyGender.Male);
translator.Add(TheirGender.Female, MyGender.Female);
translator.Add(TheirGender.Unknown, MyGender.Unknown);
return translator;
}
public static MyGender Translate(this TheirGender theirValue)
{
return GenderTranslator[theirValue];
}
public static TheirGender Translate(this MyGender myValue)
{
return GenderTranslator.FirstOrDefault(x => x.Value == myValue).Key;
}
}
ToString()を使用して最初の列挙型をその名前に変換し、次にEnum.Parse()を使用して文字列を他のEnumに変換できます。値が宛先列挙型でサポートされていない場合(つまり、「不明な」値の場合)、これは例外をスローします