Jak uzyskać wartość wyliczenia według ciągu lub int


110

Jak uzyskać wartość wyliczenia, jeśli mam ciąg wyliczenia lub wartość wyliczenia int. np .: Jeśli mam wyliczenie w następujący sposób:

public enum TestEnum
{
    Value1 = 1,
    Value2 = 2,
    Value3 = 3
}

aw jakiejś zmiennej łańcuchowej mam następującą wartość „wartość1”:

string str = "Value1" 

lub w jakiejś zmiennej int mam wartość 2 jak

int a = 2;

jak mogę uzyskać wystąpienie enum? Chcę metody ogólnej, w której mogę podać wyliczenie i mój ciąg wejściowy lub wartość int, aby uzyskać wystąpienie wyliczenia.


możliwy duplikat wartości Get enum int przez string

Odpowiedzi:


219

Nie, nie chcesz metody ogólnej. To jest dużo łatwiejsze:

MyEnum myEnum = (MyEnum)myInt;

MyEnum myEnum = (MyEnum)Enum.Parse(typeof(MyEnum), myString);

Myślę, że to też będzie szybsze.


To właściwie jest właściwy sposób zrobienia tego. Nie ma ogólnego sposobu analizowania typów z tego samego powodu, dla którego nie ma interfejsu IParsable.
Johannes

1
@Johannes Co przez to rozumiesz? Jest ogólny sposób, polecam moją odpowiedź i inne.
Sriram Sakthivel

1
@SriramSakthivel Problem, który opisuje OP, został rozwiązany tak, jak pokazała to KendallFrey. Nie można przeprowadzić analizy ogólnej - zobacz tutaj: informit.com/blogs/blog.aspx?uk=Why-no-IParseable-interface . Żadne inne rozwiązanie nie ma żadnej przewagi w porównaniu z „wbudowanym” rozwiązaniem języka C #. Maksymalne, jakie możesz mieć, to ICanSetFromString <T>, w którym tworzysz i inicjujesz obiekt do jego wartości domyślnej (T), a w następnym kroku przechodzi w reprezentatywny ciąg. Jest to bliskie odpowiedzi udzielonej przez OP - ale jest to bezcelowe, ponieważ zwykle jest to problem projektowy i pominięto większy punkt w projekcie systemu.
Johannes

Ta odpowiedź działała bardzo dobrze, szczególnie w przypadku wielu przykładów użycia int i string. Dziękuję Ci.
Termato

1
Myślę, że to teraz działa, jest trochę bardziej zwięzłe: Enum.Parse <MyEnum> (myString);
Phil B

32

Można to zrobić na wiele sposobów, ale wystarczy prosty przykład. Wystarczy go rozszerzyć o niezbędne kodowanie obronne, aby sprawdzić bezpieczeństwo typu i nieprawidłowe analizowanie itp.

    /// <summary>
    /// Extension method to return an enum value of type T for the given string.
    /// </summary>
    /// <typeparam name="T"></typeparam>
    /// <param name="value"></param>
    /// <returns></returns>
    public static T ToEnum<T>(this string value)
    {
        return (T) Enum.Parse(typeof(T), value, true);
    }

    /// <summary>
    /// Extension method to return an enum value of type T for the given int.
    /// </summary>
    /// <typeparam name="T"></typeparam>
    /// <param name="value"></param>
    /// <returns></returns>
    public static T ToEnum<T>(this int value)
    {
        var name = Enum.GetName(typeof(T), value);
        return name.ToEnum<T>();
    }

17

Może być znacznie prostsze, jeśli użyjesz metod TryParselub Parsei ToObject.

public static class EnumHelper
{
    public static  T GetEnumValue<T>(string str) where T : struct, IConvertible
    {
        Type enumType = typeof(T);
        if (!enumType.IsEnum)
        {
            throw new Exception("T must be an Enumeration type.");
        }
        T val;
        return Enum.TryParse<T>(str, true, out val) ? val : default(T);
    }

    public static T GetEnumValue<T>(int intValue) where T : struct, IConvertible
    {
        Type enumType = typeof(T);
        if (!enumType.IsEnum)
        {
            throw new Exception("T must be an Enumeration type.");
        }

        return (T)Enum.ToObject(enumType, intValue);
    }
}

Jak zauważył @chrfin w komentarzach, możesz bardzo łatwo uczynić z tego metodę rozszerzającą, dodając thisprzed typem parametru, który może być przydatny.


2
Teraz dodaj a thisdo parametru i ustaw EnumHelperstatyczne i możesz ich również używać jako rozszerzeń (zobacz moją odpowiedź, ale masz lepszy / kompletny kod do reszty) ...
Christoph Fink

@chrfin Dobry pomysł, ale nie preferuję go, ponieważ będzie pojawiał się w intelisense, gdzie nie jest to wymagane, gdy mamy przestrzenie nazw w zakresie. Myślę, że to będzie denerwujące.
Sriram Sakthivel

1
@chrfin Dzięki za komentarz, dodany jako uwaga w mojej odpowiedzi.
Sriram Sakthivel

5

Poniżej znajduje się metoda w języku C # służąca do pobierania wartości wyliczenia według ciągu

///
/// Method to get enumeration value from string value.
///
///
///

public T GetEnumValue<T>(string str) where T : struct, IConvertible
{
    if (!typeof(T).IsEnum)
    {
        throw new Exception("T must be an Enumeration type.");
    }
    T val = ((T[])Enum.GetValues(typeof(T)))[0];
    if (!string.IsNullOrEmpty(str))
    {
        foreach (T enumValue in (T[])Enum.GetValues(typeof(T)))
        {
            if (enumValue.ToString().ToUpper().Equals(str.ToUpper()))
            {
                val = enumValue;
                break;
            }
        }
    }

    return val;
}

Poniżej znajduje się metoda w C #, aby uzyskać wartość wyliczenia przez int.

///
/// Method to get enumeration value from int value.
///
///
///

public T GetEnumValue<T>(int intValue) where T : struct, IConvertible
{
    if (!typeof(T).IsEnum)
    {
        throw new Exception("T must be an Enumeration type.");
    }
    T val = ((T[])Enum.GetValues(typeof(T)))[0];

    foreach (T enumValue in (T[])Enum.GetValues(typeof(T)))
    {
        if (Convert.ToInt32(enumValue).Equals(intValue))
        {
            val = enumValue;
            break;
        }             
    }
    return val;
}

Jeśli mam wyliczenie w następujący sposób:

public enum TestEnum
{
    Value1 = 1,
    Value2 = 2,
    Value3 = 3
}

wtedy mogę skorzystać z powyższych metod, jak

TestEnum reqValue = GetEnumValue<TestEnum>("Value1");  // Output: Value1
TestEnum reqValue2 = GetEnumValue<TestEnum>(2);        // OutPut: Value2

Mam nadzieję, że to pomoże.


4
Czy możesz również podać numer referencyjny, skąd to masz?
JonH

Aby to skompilować musiałem zmodyfikować pierwszą linię na public T GetEnumValue <T> (int intValue), gdzie T: struct, IConvertible Również uważaj na dodatkowe '}' na końcu
Avi

3

Myślę, że zapomniałeś o ogólnej definicji typu:

public T GetEnumValue<T>(int intValue) where T : struct, IConvertible // <T> added

i możesz go ulepszyć, aby był jak najbardziej wygodny, na przykład:

public static T ToEnum<T>(this string enumValue) : where T : struct, IConvertible
{
    return (T)Enum.Parse(typeof(T), enumValue);
}

wtedy możesz:

TestEnum reqValue = "Value1".ToEnum<TestEnum>();

2

Spróbuj czegoś takiego

  public static TestEnum GetMyEnum(this string title)
        {    
            EnumBookType st;
            Enum.TryParse(title, out st);
            return st;          
         }

Więc możesz to zrobić

TestEnum en = "Value1".GetMyEnum();

2

Z bazy danych SQL pobierz wyliczenie takie jak:

SqlDataReader dr = selectCmd.ExecuteReader();
while (dr.Read()) {
   EnumType et = (EnumType)Enum.Parse(typeof(EnumType), dr.GetString(0));
   ....         
}

2

Po prostu spróbuj tego

To inny sposób

public enum CaseOriginCode
{
    Web = 0,
    Email = 1,
    Telefoon = 2
}

public void setCaseOriginCode(string CaseOriginCode)
{
    int caseOriginCode = (int)(CaseOriginCode)Enum.Parse(typeof(CaseOriginCode), CaseOriginCode);
}

0

Oto przykład pobierania ciągu / wartości

    public enum Suit
    {
        Spades = 0x10,
        Hearts = 0x11,
        Clubs = 0x12,
        Diamonds = 0x13
    }

    private void print_suit()
    {
        foreach (var _suit in Enum.GetValues(typeof(Suit)))
        {
            int suitValue = (byte)(Suit)Enum.Parse(typeof(Suit), _suit.ToString());
            MessageBox.Show(_suit.ToString() + " value is 0x" + suitValue.ToString("X2"));
        }
    }

    Result of Message Boxes
    Spade value is 0x10
    Hearts value is 0x11
    Clubs value is 0x12
    Diamonds value is 0x13

0

Możesz użyć następującej metody, aby to zrobić:

public static Output GetEnumItem<Output, Input>(Input input)
    {
        //Output type checking...
        if (typeof(Output).BaseType != typeof(Enum))
            throw new Exception("Exception message...");

        //Input type checking: string type
        if (typeof(Input) == typeof(string))
            return (Output)Enum.Parse(typeof(Output), (dynamic)input);

        //Input type checking: Integer type
        if (typeof(Input) == typeof(Int16) ||
            typeof(Input) == typeof(Int32) ||
            typeof(Input) == typeof(Int64))

            return (Output)(dynamic)input;

        throw new Exception("Exception message...");
    }

Uwaga: ta metoda jest tylko próbką i możesz ją ulepszyć.

Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.