DEV Community

Cover image for 5 MORE things about enums in C#
Davide Bellone
Davide Bellone

Posted on • Originally published at code4it.dev on

5 MORE things about enums in C#

In a previous article, I explained some details about enums in C#. Here I'll talk about some other things that are useful and/or curious to know about them.

#1: Define enum internal type

If you remember, enums can be int, but also short, uint and so on. How can you define which type to use?

It's simple, you can add it after the declaration of the enum:

public enum PetTypes : ulong
{
    Cat = 2,
    Dog = 3,
    Alligator = 4
}
Enter fullscreen mode Exit fullscreen mode

and, if we look at the generated IL, we'll see the result:

IL result for uint enum

The fact that you can define them as int and as uint is a hint (pun intended) that you can also use negative numbers as inner value:

public enum Time
{
    Past = -1,
    Present = 0,
    Future = 1
}
Enter fullscreen mode Exit fullscreen mode

#2: Enums combination within the definition

If you set each element's value as a power of 2, like you did for Flags

enum Beverage
{
    Water = 1,
    Beer = 2,
    Tea = 4,
    RedWine = 8,
    WhiteWine = 16
}
Enter fullscreen mode Exit fullscreen mode

you can write something like

var beverage = Beverage.RedWine | Beverage.WhiteWine;

if(beverage.HasFlag(Beverage.RedWine) || beverage.HasFlag(Beverage.WhiteWine)){
    Console.WriteLine("This is wine");
}
Enter fullscreen mode Exit fullscreen mode

Now imagine that you have to check many times if a beverage is a wine: you can repeat the check, or extract it to a separate method.

Or you can add a value to the enum:

[Flags]
enum Beverage
{
    Water = 1,
    Beer = 2,
    Tea = 4,
    RedWine = 8,
    WhiteWine = 16,
    Wine = RedWine | WhiteWine
}
Enter fullscreen mode Exit fullscreen mode

This simplifies your code to

var beverage = Beverage.RedWine | Beverage.WhiteWine;

if(beverage.HasFlag(Beverage.Wine)){
    Console.WriteLine("This is wine");
}
Enter fullscreen mode Exit fullscreen mode

It's a small trick, but it can help you clean your code.

#3: Serializer

Now it's time to use enums for real-word scenarios: maybe a .NET Core 3 API?

I have created a simple API controller that returns a single movie:

[Route("api/[controller]")]
[ApiController]
public class MoviesController : ControllerBase
{
    [HttpGet]
    public ActionResult<Movie> Get()
    {
        var movie = new Movie()
        {
            Name = "My action movie",
            ReleaseDate = new DateTime(2019, 3, 16),
            Genre = MovieGenre.Action
        };

        return movie;
    }
}
Enter fullscreen mode Exit fullscreen mode

What will this endpoint return? As you've already learned, internally an enum is nothing but a number, so the returned JSON will be this one:

{
  "name": "My action movie",
  "releaseDate": "2019-03-16T00:00:00",
  "genre": 23
}
Enter fullscreen mode Exit fullscreen mode

Yes, I know, you were expecting the string value.

But you can have it by updating the Startup class:

public void ConfigureServices(IServiceCollection services)
{
    services.AddControllers().AddJsonOptions(o =>
    {
        o.JsonSerializerOptions.Converters.Add(new JsonStringEnumConverter());
    });
}
Enter fullscreen mode Exit fullscreen mode

or, if you want to set a serializer only to a specific enum, you can add an attribute to it:

+   [JsonConverter(typeof(JsonStringEnumConverter))]
    public enum MovieGenre
    {
Enter fullscreen mode Exit fullscreen mode

#4: The real meaning of the Flags attribute

Thanks to Insulind on Reddit I found out that actually the [Flags] attribute's job is not to allow multiple values to be stored in a single field, rather to have a better string representation.

If we remove the Flags attribute on the Beverage enum, we can assign multiple values as if it had the attribute enabled.

enum Beverage
{
    Water = 1,
    Beer = 2,
    Tea = 4,
    RedWine = 8,
    WhiteWine = 16
}

// and, in a method

var beverage = Beverage.Water| Beverage.RedWine;
Enter fullscreen mode Exit fullscreen mode

Everything works, even without the HasFlag method. The difference comes if we get the string value of that variable: now it returns 9, because it's getting directly the numeric value.

If we put the Flags attribute, everything changes: the string value will be Water, RedWine, so the comma-separated list of their values.

This makes sense only if you use multiple values: as we've already seen, if you print a single value you'll get the string value of it.

If you remember when I talked about how to format an enum, you can use ToString("g") and ToString("f") to get the enum name. I specified that there is a small difference but I haven't explained what was it about.

If you set a combined enum value but you don't add the Flags attribute, you'll end up with different results:

    var beverage = Beverage.Water | Beverage.RedWine;
    beverage.ToString(); //9
    beverage.ToString("g"); //9
    beverage.ToString("f"); //Water, RedWine
Enter fullscreen mode Exit fullscreen mode

On the contrary, if you add the Flags attribute, they all return Water, RedWine.

#5 Flags best practices

As always, there are some best practices you should follow. The following ones are about the usage of Enums with the Flags attribute.

  1. Use only powers of two when defining flags, so that you won't overlap values when combining more enums; this will give you also the possibility to use bitwise operators, like OR, AND and XOR.
  2. You should add a None value, and set it to 0, to represent the lack of value. Of course, consider that HasFlag(EnumName.None) will always return true.
  3. In older versions of .NET, the HasFlag method was less efficient than a bitwise AND check. So, to check if a value contains Beverage.Water, you could do ((beverage & Beverage.Water) == Beverage.Water). With newer versions this problem has been fixed.
  4. As always, remember to validate input parameters, just like you should do for simple enums.

For more, you can refer to Microsoft documentation.

Wrapping up

Here we've seen more things to know about enums. Probably you won't use all of these capabilities in real-world scenarios, but I think you should know that they exist.

To recap, we've seen that

  • you can change the underlying type to short, uint and similar;
  • you can also define negative values;
  • inside the enum definition, you can create values that represent the union of other two (or more);
  • by default, .NET APIs return enum values as numbers; to fix it, you must add a configuration in your Startup class;
  • you can also omit the Flags attribute; the downside is that when you print the value of a variable that is the union of 2 enums, the returned string is the sum of the enums stored inside that variable, not the string representation.

Do you know something to add?

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!

Top comments (0)