Why does Enum.GetValues() return names when using “var”?

前端 未结 7 1056
囚心锁ツ
囚心锁ツ 2021-01-17 07:44

Can anyone explain this?

alt text http://www.deviantsart.com/upload/g4knqc.png

using System;

namespace TestEnum2342394834
{
    class Program
    {
         


        
7条回答
  •  天涯浪人
    2021-01-17 08:00

    EDIT: added some sample code that explores many (perhaps all?) possible ways of iterating over the array.

    Enum types are considered to be "derived" from int by default. You can choose to derive it from one of the other integer types if you want, such as byte, short, long, etc.

    In both cases, the call to Enum.GetValues is returning an array of ReportStatus objects.

    Using the var keyword in the first loop tells the compiler to use the specified type of the array, which is ReportStatus, to determine the type of the value variable. The ToString implementation for enums is to return the name of the enum entry, not the integer value it represents, which is why the names are being output from the first loop.

    Using an int variable in the second loop causes the values returned by Enum.GetValues to be implicitly converted from ReportStatus to int. Calling ToString on an int will, of course, return a string representing the integer value. The implicit conversion is what causes the difference in behavior.

    UPDATE: As others have pointed out, the Enum.GetValues function returns an object typed as Array, and as a result it is an enumerable of Object types, not ReportStatus types.

    Regardless, the end result is the same whether iterating over Array or ReportStatus[]:

    class Program
    {
        enum ReportStatus
        {
            Assigned = 1,
            Analyzed = 2,
            Written = 3,
            Reviewed = 4,
            Finished = 5,
        }
    
        static void Main(string[] args)
        {
            WriteValues(Enum.GetValues(typeof(ReportStatus)));
    
            ReportStatus[] values = new ReportStatus[] {
                ReportStatus.Assigned,
                ReportStatus.Analyzed,
                ReportStatus.Written,
                ReportStatus.Reviewed,
                ReportStatus.Finished,
            };
    
            WriteValues(values);
        }
    
        static void WriteValues(Array values)
        {
            foreach (var value in values)
            {
                Console.WriteLine(value);
            }
    
            foreach (int value in values)
            {
                Console.WriteLine(value);
            }
        }
    
        static void WriteValues(ReportStatus[] values)
        {
            foreach (var value in values)
            {
                Console.WriteLine(value);
            }
    
            foreach (int value in values)
            {
                Console.WriteLine(value);
            }
        }
    }
    

    Just for some extra fun, I've added some code below demonstrating several different ways of iterating over the specified array with a foreach loop, including comments that describes in detail what's going on in each case.

    class Program
    {
        enum ReportStatus
        {
            Assigned = 1,
            Analyzed = 2,
            Written = 3,
            Reviewed = 4,
            Finished = 5,
        }
    
        static void Main(string[] args)
        {
            Array values = Enum.GetValues(typeof(ReportStatus));
    
            Console.WriteLine("Type of array: {0}", values.GetType().FullName);
    
            // Case 1: iterating over values as System.Array, loop variable is of type System.Object
            // The foreach loop uses an IEnumerator obtained from System.Array.
            // The IEnumerator's Current property uses the System.Array.GetValue method to retrieve the current value, which uses the TypedReference.InternalToObject function.
            // The value variable is passed to Console.WriteLine(System.Object).
            // Summary: 0 box operations, 0 unbox operations, 1 usage of TypedReference
            Console.WriteLine("foreach (object value in values)");
            foreach (object value in values)
            {
                Console.WriteLine(value);
            }
    
            // Case 2: iterating over values as System.Array, loop variable is of type ReportStatus
            // The foreach loop uses an IEnumerator obtained from System.Array.
            // The IEnumerator's Current property uses the System.Array.GetValue method to retrieve the current value, which uses the TypedReference.InternalToObject function.
            // The current value is immediatly unboxed as ReportStatus to be assigned to the loop variable, value.
            // The value variable is then boxed again so that it can be passed to Console.WriteLine(System.Object).
            // Summary: 1 box operation, 1 unbox operation, 1 usage of TypedReference
            Console.WriteLine("foreach (ReportStatus value in values)");
            foreach (ReportStatus value in values)
            {
                Console.WriteLine(value);
            }
    
            // Case 3: iterating over values as System.Array, loop variable is of type System.Int32.
            // The foreach loop uses an IEnumerator obtained from System.Array.
            // The IEnumerator's Current property uses the System.Array.GetValue method to retrieve the current value, which uses the TypedReference.InternalToObject function.
            // The current value is immediatly unboxed as System.Int32 to be assigned to the loop variable, value.
            // The value variable is passed to Console.WriteLine(System.Int32).
            // Summary: 0 box operations, 1 unbox operation, 1 usage of TypedReference
            Console.WriteLine("foreach (int value in values)");
            foreach (int value in values)
            {
                Console.WriteLine(value);
            }
    
            // Case 4: iterating over values as ReportStatus[], loop variable is of type System.Object.
            // The foreach loop is compiled as a simple for loop; it does not use an enumerator.
            // On each iteration, the current element of the array is assigned to the loop variable, value.
            // At that time, the current ReportStatus value is boxed as System.Object.
            // The value variable is passed to Console.WriteLine(System.Object).
            // Summary: 1 box operation, 0 unbox operations
            Console.WriteLine("foreach (object value in (ReportStatus[])values)");
            foreach (object value in (ReportStatus[])values)
            {
                Console.WriteLine(value);
            }
    
            // Case 5: iterating over values as ReportStatus[], loop variable is of type ReportStatus.
            // The foreach loop is compiled as a simple for loop; it does not use an enumerator.
            // On each iteration, the current element of the array is assigned to the loop variable, value.
            // The value variable is then boxed so that it can be passed to Console.WriteLine(System.Object).
            // Summary: 1 box operation, 0 unbox operations
            Console.WriteLine("foreach (ReportStatus value in (ReportStatus[])values)");
            foreach (ReportStatus value in (ReportStatus[])values)
            {
                Console.WriteLine(value);
            }
    
            // Case 6: iterating over values as ReportStatus[], loop variable is of type System.Int32.
            // The foreach loop is compiled as a simple for loop; it does not use an enumerator.
            // On each iteration, the current element of the array is assigned to the loop variable, value.
            // The value variable is passed to Console.WriteLine(System.Int32).
            // Summary: 0 box operations, 0 unbox operations
            Console.WriteLine("foreach (int value in (ReportStatus[])values)");
            foreach (int value in (ReportStatus[])values)
            {
                Console.WriteLine(value);
            }
    
            // Case 7: The compiler evaluates var to System.Object.  This is equivalent to case #1.
            Console.WriteLine("foreach (var value in values)");
            foreach (var value in values)
            {
                Console.WriteLine(value);
            }
    
            // Case 8: The compiler evaluates var to ReportStatus.  This is equivalent to case #5.
            Console.WriteLine("foreach (var value in (ReportStatus[])values)");
            foreach (var value in (ReportStatus[])values)
            {
                Console.WriteLine(value);
            }
        }
    }
    

    -- Updated my comments in the sample above; upon double-checking I saw that the System.Array.GetValue method actually uses the TypedReference class in order to extract an element of the array and return it as System.Object. I had originally written that there was a boxing operation happening there, but that is technically not the case. I'm unsure what the comparison of a box operation is vs. a call to TypedReference.InternalToObject; I assume it depends on the CLR implementation. Regardless, I believe the details are more or less correct now.

提交回复
热议问题