5 things you didn't know about enums in C#

bellonedavide

Davide Bellone

Posted on May 8, 2020

5 things you didn't know about enums in C#

Enums are really useful if you want to, well, enumerate the possible values for a field. An example of enumeration is the list of movie genres:

public enum MovieGenre
{
    Action,
    Comedy,
    Drama,
    Musical,
    Thriller,
    Horror
}
Enter fullscreen mode Exit fullscreen mode

This way, you can define that a field accepts only values coming from that specific enum, thus narrowing down the list of possible values:

public class Movie
{
    public string Name { get; set; }
    public DateTime ReleaseDate { get; set; }
    public MovieGenre Genre { get; set; }
}
Enter fullscreen mode Exit fullscreen mode

So, now, you can instantiate a new movie as

new Movie()
{
    Name = "My movie",
    ReleaseDate = DateTime.Now,
    Genre = MovieGenre.Drama
};
Enter fullscreen mode Exit fullscreen mode

Simple, right?

As usual, there is more than this, even if we are used to staying on the surface.

#1: Enums are nothing but masked numbers

Even though their structure may let you think of a complex structure, enums are value types.
This means that if you pass an enum to a function, you will not modify it as if it were an object.

In fact, if we define a function that sets a given MovieGenre to Musical:

static void SetToMusical(MovieGenre genre)
{
    genre = MovieGenre.Musical;
}
Enter fullscreen mode Exit fullscreen mode

and we try to update a value, we will notice that we haven't actually updated the value.

MovieGenre genre = MovieGenre.Action;
Console.WriteLine(genre);// Action

SetToMusical(genre);
Console.WriteLine(genre);// Action
Enter fullscreen mode Exit fullscreen mode

Internally, an enum is a numeric type: it can be made of byte, sbyte, short, ushort, int, uint, long, or ulong values.
By default, an enum is a static, Int32 value, whose first element has value 0 and all the following elements have their value increased by 1 compared to the previous one.

Enums as Intermediate Language

We can customize this default behavior by using a different starting value and then have all the other values will follow the numeric order.

public enum MovieGenre
{
    Action = 23,
    Comedy,
    Drama,
    Musical,
    Thriller,
    Horror
}
Enter fullscreen mode Exit fullscreen mode

Enums as IL starting with a different value

This means that if you reorder the elements (for example, because you want the values to be in alphabetic order, so moving Horror between Drama and Musical), you might mess up all the following values and bring subtle bugs in.

How to fix it? Simple: specify a value for each element!

public enum MovieGenre
{
    Action = 23,
    Comedy = 24,
    Drama = 25,
    Musical = 26,
    Thriller = 27,
    Horror = 28
}
Enter fullscreen mode Exit fullscreen mode

This way you can reorder the items without the risk of creating bugs.

#2: Many ways to get the value

Even if internally an enum is nothing but a number, you can get its friendly name. There are many ways to do that.

If you want the string value of a statically defined enum value, you can rely on the nameof operator, with

nameof(MovieGenre.Comedy);
Enter fullscreen mode Exit fullscreen mode

or, using the Enums class, try with the GetName method:

Enum.GetName(typeof(MovieGenre), MovieGenre.Comedy);
Enter fullscreen mode Exit fullscreen mode

The GetName method is a bit cumbersome to write, so probably you'd prefer using the nameof operator, which was released with C# 6.

How can you get the value if it is set at runtime?

Of course, you may use the ToString method.

MovieGenre mg = MovieGenre.Musical;
var musicalValueAsString = mg.ToString();
Enter fullscreen mode Exit fullscreen mode

Or, if you want to rely on the Enum class, you can try with

Enum.Format(typeof(MovieGenre), mg, "g")
Enter fullscreen mode Exit fullscreen mode

Have you noticed the last parameter of the previous snippet? What does it mean?

Well, as you can format Guids, you can use flags to format an enum as well.

  • G and F return the name of the enum. There is a small difference that occurs when an enum is used with the flag attribute (I'll talk about it later)
  • D represents the value in decimal form
  • X represents the value in hexadecimal form.

These flags can be used both on the Enum.Format and the ToString method.

var action = MovieGenre.Action; // remember, its value is 23

action.ToString("g"); // "Action"
action.ToString("x"); // "00000017"
action.ToString("d"); // "23"
Enter fullscreen mode Exit fullscreen mode

You can read more details on those values on Microsoft documentation.

If you need the numeric value of the enum, you can simply cast it.

(int)MovieGenre.Action
Enter fullscreen mode Exit fullscreen mode

#3: Parsing and casting enums

You can transform an enum to a string, and of course you can do the opposite!

MovieGenre e;
Enum.TryParse<MovieGenre>("Action", out e);
Enter fullscreen mode Exit fullscreen mode

or, if you prefer defining the variable inline,

Enum.TryParse<MovieGenre>("Action", out MovieGenre e);
Enter fullscreen mode Exit fullscreen mode

neat and clean, isn't it?

A thing that you must keep in mind is that, if the string you pass to the method does not exist, TryParse will set to 0 the out variable: this can cause bugs if you have defined a value associated with 0.

public enum Status
{
    OK = 0,
    Failed = 1,
    Waiting = 2
}

// and, within a method...

Enum.TryParse<Status>("Feiled", out Status st); // OK
Enter fullscreen mode Exit fullscreen mode

In the above example, "Feiled" is not a valid value, so the assigned value is 0 which is casted to OK.

You can prevent this bug in 2 ways: by checking on the returned value of TryParse, which returns true if the parsing was successful, false otherwise; or you can add an additional check before the parsing, using the IsDefined method, with something like Enum.IsDefined(typeof(Status), "Feiled").

#4: Flagged enums

What if an enum fields must allow multiple values? After all, a movie can have more than one genre, right?

You could implement it as a list (or an array) of flags, or... you can use the Flags attribute.
This flag allows to easily apply OR operations on enums, making the code cleaner and more readable. The downside is that now enums values can't have custom values, but must be a power of 2, so 1, 2, 4, 8 and so on.

[Flags]
public enum MovieGenre
{
    Action = 1,
    Comedy = 2,
    Drama = 4,
    Musical = 8,
    Thriller = 16,
    Horror = 32
}
Enter fullscreen mode Exit fullscreen mode

So now we can create an action-comedy movie

var movie = new Movie()
{
    Name = "Bad Boys",
    ReleaseDate = new DateTime(1995, 4, 7),
    Genre = MovieGenre.Action | MovieGenre.Comedy
};
Enter fullscreen mode Exit fullscreen mode

Now that you have flags on enums, whatcha gonna do?
You can use the HasFlag method to, well, check if a value has a certain flag

MovieGenre mg = MovieGenre.Action | MovieGenre.Comedy;

if (mg.HasFlag(MovieGenre.Comedy))
{
    // Do something
}
Enter fullscreen mode Exit fullscreen mode

This is more performant than looping though a list of enums, since now we're working directly on bits.

#5: Enum best practices

As always, there are some best practices to follow. The following ones are suggested directly on the Microsoft documentation:

  1. If you have a default value for the enumeration, set its value to 0;
  2. If there isn't an obvious default value, create a value (set to 0) that represents the fallback case (for example, create a None value and set it to 0);
  3. Validate inputs for methods that allow enums as parameters, since enums are nothing but numbers, so a simple cast can cause unexpected results;

Let me explain the third point: do you remember the Status enum?

Here's a method that tells if the input is valid:

string printValidity(Status status){
    switch (status)
    {
        case Status.Failed:  
        case Status.OK:  
        case Status.Waiting:
            return "Valid input";
        default:
            return "Invalid input";
    }
}
Enter fullscreen mode Exit fullscreen mode

and well, you can imagine how it works.

What happens if you do this?

var validity = printValidity((Status) 1234);
Enter fullscreen mode Exit fullscreen mode

Exactly, the value is Invalid input. So, remember to validate inputs!

Conclusion

In this article, we've seen that

  • enums are just numbers in disguise;
  • you can format an enum as a string, a hexadecimal value or a numeric value;
  • you can use flags to define multiple values;
  • you should follow best practices: remember to define a default value and to validate inputs;

Happy coding!


Originally published at code4it.dev.

📝📝📝

I write about C#, .NET, Azure, and everything around them. Check it out!

🐧🐧🐧

Let's keep in touch on Twitter!

💖 💪 🙅 🚩
bellonedavide
Davide Bellone

Posted on May 8, 2020

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related

Demystifying Algorithms: Doubly Linked Lists
datastructures Demystifying Algorithms: Doubly Linked Lists

November 29, 2024

Demystifying Algorithms: Circular Linked Lists
datastructures Demystifying Algorithms: Circular Linked Lists

November 29, 2024

Demystifying Algorithms: Singly Linked Lists
datastructures Demystifying Algorithms: Singly Linked Lists

November 29, 2024

i18n e ASP.NET Core Web API
csharp i18n e ASP.NET Core Web API

November 28, 2024