Quantcast
Channel: Telerik Blogs
Viewing all articles
Browse latest Browse all 5210

What I Like About C# 9

$
0
0

We’ll start with how I’ve been using some of the new features of C# 9 and why I like them. Then we’ll look at an example that uses a Person class and expands on it. At the end, I may even hint at C# 10 features.

I’ve been a software engineer for 20+ years, and as the adage goes, You can’t teach an old dog new tricks. However, if there is one thing I learned in those 20+ years, it’s that I am ALWAYS learning. There are always new technologies coming out, new languages and new products to solve complex problems. .NET 5 introduced C# 9, which had many new language features. So it was time for me to learn some new tricks, and I dove into .NET 5’s C# 9 language additions.

After using these new language features, keywords and syntax, I noticed that they started to save me keystrokes and time. Since these language additions helped me, I wanted to share them with you.

Let’s take a look at some of the new language features.

Records

The new record keyword defines a reference type that provides some built-in functionality for representing data. You might be thinking that this sounds a lot like a class, and you would be correct—it does. However, the intent is to provide smaller and more concise types to represent immutable data. I like to think of them as a type used primarily to transfer data and not have a lot of methods or data manipulation.

More on C# 9 records.

Defining a Record

There are a few different ways to define a record. The simplest form is:

public record Person(string FirstName, string LastName);

At first glance, at least for me, that seemed weird. It has a method look and feel. There is even a semicolon at the end. But, the above line creates a Person type with the read/write properties of FirstName and LastName. You can access the Person as follows:

var person = new Person("Joseph", "Guadagno");
Console.WriteLine(person.FirstName); // Outputs Joseph
Console.WriteLine(person.LastName); // Outputs Guadagno

So far, this looks very class-like. Well, it is, except for the declaration. We already saved a bunch of keystrokes. But let’s dig more into it.

Another way to define the Person record is more class-like:

public record Person
{
    public string FirstName { get; set;}
    public string LastName { get; set;}
}

Creating Records by Position

You can further reduce some typing and remove some boilerplate code using the new positional syntax for records. For example, if you wanted to declare a variable with the class approach and initialize it with data, you would do something like this.

var person = new Person { FirstName = "Joseph", LastName="Guadagno"};

With positional syntax, that would look like this.

Person person = new ("Joseph", "Guadagno");

That’s 26 fewer characters. Behind the scenes the compiling is creating a lot of the boilerplate code for you. The compiler creates a constructor that matches the position of the record declaration. Since the FirstName property was the first property declared when we defined the method, it assumes that the Joseph value should be the value of the FirstName property. The compiler also generated all the properties as init-only (more on that later), meaning the properties cannot get set after initialization, making them read-only.

Value Equality

One set of built-in functionality that records provide is value equality. When checking to see if two records are equal, it will look at the values of each of the properties and not the reference.

Assuming the definition of:

public record Person(string FirstName, string LastName);

when comparing records:

Person person1 = new ("Joseph", "Guadagno");
Person person2 = new ("Joseph", "Guadagno");

Console.WriteLine(person1 == person2); // outputs True
Console.WriteLine(ReferenceEquals(person1, person2)); // outputs False

Since person2 has the same FirstName and LastName of person2, they are equal, although the references are not.

Improved ToString()

Using the record keyword gets you another built-in method. What a deal! An improved ToString method. I really wish this was opt-in standard for classes too.

The ToString method outputs the following format.

<record type name> { <property name> = <value>, <property name> = <value>, ...}

For a record defined as:

public record Person(string FirstName, string LastName);

and initialized as:

Person person = new {"Joseph", "Guadagno"};

the ToString method would output a string like:

Person { FirstName = Joseph, LastName = Guadagno }

If there is a reference type as one of the properties of the record, the record’s ToString implementation will output the type name of it.

NOTE: Don’t try to use the ToString method to determine the records properties.

Inheriting Records

Records can be inherited the same way classes are except for the following:

  • Records can’t inherit from a class.
  • Class can’t inherit from a record.
  • When comparing records, the type of record is used as part of the comparison and not just the values.

Copying Records

Copying records is pretty easy. As an added bonus, the syntax makes the code easier to read.

Let’s say we have a Person record defined as:

public record Person
{
    string FirstName { get; set;}
    string LastName { get; set;}
    string HomeState { get; set;}
}

Let’s also say we want to create one Person and make multiple copies and just change a few properties, as if we were creating variables for the whole family. In our case, the LastName and HomeState properties are the same and using records along with the with keyword makes this easier.

var me = new Person("Joseph", "Guadagno", "Arizona");
var wife = me with {FirstName = "Deidre"};
var son = me with {FirstName = "Joseph Jr."};
var daughter = me with {FirstName = "Emily"};

Now, the wife, son and daughter objects have the property of LastName set to Guadagno and HomeState set to Arizona.

Defining Set-Once Properties

You can also use the new init keyword to make certain properties settable on initialization only. The init keyword works with properties or indexers in struct, class or record.

Let’s say with want to define a Person record with FirstName, LastName and CreateOnDate properties. The CreatedOnDate should not be editable after the record is initialized. We would declare the record like this.

public record Person 
{
    public string FirstName { get; set;}
    public string LastName { get; set;}
    public DateTime CreatedOnDate { get; init;}
}

You see on line 5 we have the keyword init instead of set. This means the CreatedOnDate can only be set when initialized.

var person = new Person("Joseph", "Guadagno", DateTime.Now());

After declaring this record, we are limited as to what properties we can change.

person.FirstName = "Joe"; // valid
person.CreatedOnDate = DateTime.Now(); // You will get a compile error

Line 2 will cause a compilation error because the property CreatedOnDate was set to init only.

Alternative Declaration

You can also declare the setter of a property with a backing field as init only.

public class Person
{
    private readonly DateTime _dateOfBirth;
    public DateTime DateOfBirth 
    {
        get => _dateOfBirth;
        init => (value ?? throw new ArgumentNullException(nameof(DateOfBirth)));
    }
    public string FirstName { get; set;}
    public string LastName { get; set;}
}

On line 7, we define the class Person with an init-only property DateOfBirth that must be set at initialization or you will get a compile error or runtime exception depending on the implementation.

This is valid (assuming the definition above):

var person = new Person{FirstName="Joseph", LastName="Guadagno", DateOfBirth=DateTime.Now()};

This is not (assuming the definition above):

var person = new Person{FirstName="Joseph", LastName="Guadagno"};

Based on the above definition, this code sample will throw a runtime exception.

Top-Level Statements

I started out this post introducing the notion that C# 9’s language features help you be more productive and reduce keystrokes. Top-level statements are another one of the features.

To be honest, you probably won’t use this feature a lot. In fact, you can only have one file in your application that uses this feature. It’s generally helpful for demonstrating some functionality and removing all of the extra ceremony around the application startup. I see myself using it when I am creating presentations.

Let’s take the typical “Hello World” sample.

using System;

namespace CSharp9Features.ConsoleApp
{
    static class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine("Hello World!");
        }
    }
}

It’s 12 lines long using the default .NET C# console app template. Now with top-level statements, this can be reduced to:

System.Console.WriteLine("Hello World");

Now we reduced the code from 12 lines and 210 characters to 1 line and 40 characters.

Behind the scenes, the compiler essentially created the 12 lines and 210 characters for us. But again, C# 9 is trying to make things easier for you, so why type those lines when the compiler knows that is what you want?

In a more “realistic” example, let’s say for an ASP.NET Core Web API project, the typical template would have a Program.cs file that looks something like this:

using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;

namespace Contacts.Api
{
    public class Program
    {
        public static void Main(string[] args)
        {
            CreateHostBuilder(args).Build().Run();
        }

        public static IHostBuilder CreateHostBuilder(string[] args) =>
            Host.CreateDefaultBuilder(args)
                .ConfigureWebHostDefaults(webBuilder => { webBuilder.UseStartup<Startup>(); })
                .ConfigureLogging(logging =>
                {
                    logging.ClearProviders();
                    logging.SetMinimumLevel(LogLevel.Trace);
                });
    }
}

Now with C# 9, we can remove some of the noise and ceremony and have our code just be what the API needs to start:

using Contacts.Api;
using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;

CreateHostBuilder(args).Build().Run();

IHostBuilder CreateHostBuilder(string[] args) =>
Host.CreateDefaultBuilder(args)
    .ConfigureWebHostDefaults(webBuilder => { webBuilder.UseStartup<Startup>(); })
    .ConfigureLogging(logging =>
    {
        logging.ClearProviders();
        logging.SetMinimumLevel(LogLevel.Trace);
    });

This code now clearly states what the intent of the program.cs is without the extra namespace or Main method.

New Pattern Matching

While pattern matching is not new in C# 9, C# 9 did add a few more patterns.

Logical patterns:

  • and
  • or
  • not

Relations patterns:

  • <
  • >
  • <=
  • >=

These patterns help add readability to code. My favorite addition to this is the not pattern matcher. Now we can take all the instances of:

if (!person is null)

and make them more readable with:

if (person is not null)

While this one is more keystrokes, the extra couple of characters make it more readable to me than the ! operator.

Omitting the Type

The compiler is getting smarter. It’s not necessarily getting more intelligent, but getting better at understanding what you are trying to do and, again, reducing the keystrokes.

The C# 9 feature of target-typed new expressions demonstrates that the compiler is getting smarter. Now, based on the variable declaration or method signature, you can omit the type in variable declarations or usage.

Here we are declaring a variable _people of type List<Person>:

private List<Person> _people = new();

We no longer have to initialize the variable of _people with new List<Person>(). The compiler can assume that we want a new List of Person.

The same goes for methods. In the sample below, the method CalculateSalary expects a parameter of type PerformanceRating.

public Person CalculateSalary(PerformanceRating rating) 
{  
    // Omitted
}

If we wanted to initialize a new PerformanceRating object for the method without creating a variable, we can now.

var person = person.CalculateSalary(new ());

Or, we can pass in a new PerformanceRating object with one or more of its properties initialized.

var person = person.CalculateSalary(new () {Rating ="Rock Star"});

This syntax does take some getting used to. I think in the long run it leads to code that is easier to use. However, it might add more fuel to the var vs. typed variable declaration debate. :)

Wrap-up

Wow, that was a lot. C#9 added Record Types, Init Only setters, Top-Level programs, enhancements to pattern matching, and more.

I hope you take some time and play around with these new language features. Doing so will reduce your keystrokes and help your code to be readable in the long run.

Bonus: Coming Soon—C# 10

While not set in stone … As of the writing of this post, .NET 6 Preview 5 is planning on adding the following to C# 10.

  • Allow const interpolated strings.
  • Record types can seal ToString().
  • Allow both assignment and declaration in the same deconstruction.
  • Allow AsyncMethodBuilder attribute on methods.

For more, check out “What’s new in C# 10.0.”


Viewing all articles
Browse latest Browse all 5210

Trending Articles