Monday, July 22, 2019

What's New in C# 8.0

After 18 years of existence, C# is still as young and "sharp" as ever. Indeed, by open-sourcing its compiler and design, Microsoft strengthened the position of C# in the market by making it available to all developpers on GitHub.
Contributing and voting to set the language roadmap are the main contributors to its success in the last few years.
Last month, the C# language team in Microsoft announced the release of the preview version of VS 2019 and .NET Core 3.0.
The exciting things about it is that this version includes the new C# 8 that we can play with to test some of the new features.
In this article, I'll walk you through some of the new features introduced in the 8th version of C# and show some use cases.

Ranges and Indices

C# 8 introduces two new types and operators for collections manipulation and indexing. Basically we will have a more interesting and elegant way to index and slide collections.
New types: System.Index and System.Range.
New Operators : .. and ^.
Let's see some examples:
Index d1 = 2;  // number 2 from beginning
Index d2 = ^3; // number 3 from end
string[] week = { "Monday",
               "Tuesday",
               "Wednesday",
               "Thursday",
               "Friday",
               "Saturday",
               "Sunday" };
Console.WriteLine($"{week[d1]}, {week[d2]}"); // "Wednesday , Friday "
The element selection is 0-based if you are counting from the beginning and 1-based if you are counting from the end.
Image title

  • Get the last element:
//Old Style
var lastDay = week[week.Count -1]; // Sanday
//New Style
var lastDay = week[^1]; // Sanday
  • Get a range of elements:
//Old Style
var days = week.ToList().GetRange(2,3); // Wednesday,Thursday,Friday
//or
var days = week.Skip(2).Take(3); // Wednesday,Thursday,Friday
//New Style
var days = week[1..^2]; // Wednesday,Thursday,Friday
// [1..^2] mean 
//  1 : skip elements from the begining until the index 1
// ^2 : skip 2 element from the end
Limitation:
This new feature will be available in any framework implementing .NET Standard 2.1. For example, it will be available in .NET Core 3.0 but not in the .NET framework 4.8. If you want more informations about .NET Standard you can check this article.

Default Implementations of Interface Members

For this part, we will imagine that we have a three year old application with a bunch of classes and interfaces, and that 30 of these classes them implement an Interface called IPrintable.
This interface includes a signature for each printer operation.
Let's say you want to add a new method that does exactly the same thing as Print, but which has a different signature. In this case, there is no way around reimplementing the new method for all 30 classes.
With C# 8, you can define a default implementation for the new interface member.
Let's check how this works.
interface IPrintable
{
    void Print(string header, string text);
    // New overloadvoid Print(Document doc) => Print(doc.Header, doc.Text); 
}
class Printer : IPrintable
{
    public void Print(string header, string text) { ... }
    // Print(Document) gets default implementation
}

Switch Expressions

We can say that this is a light or short version of a switch statement with a new syntax using the => operator.
var volume = shape switch 
{
    Cube c      => Math.Pow(length,3),
    Prisme p => p.Width * p.Height * p.length,
    Cylinder cy    => Math.PI * Mathf.Pow(cy.Radius,2) * cy.heigth,
    _           => throw new UnknownShapeException(shape)
};

New Target-Typed Expressions

To build a collection with typed objects we are obliged to specify the type for each element or use the new statement for the object construction.
This is an example:
Month[] months = {new Month("January",31),new Month("April",30)};
With the new target-typed expression there's no need to specify the object type or class name we can do like this:
Month[] months = {new("January",31),new("April",30),new("June",31)};

Async Streams

C# 5 made it possible to have some asynchronous features like awaiting a result with await and async. In C# 8, we get an evolution of this feature adding the IAsyncEnumerable which allows us to yield results asynchronously.
This feature can be very useful if you are developing an IoT app and you are dealing with a lot of asynchronous calls returning data.
Here is an example:
 static async IAsyncEnumerable<string> GetElementsAsync()
{
   await Task.Delay(2000);
   yield return new Element();
}
// we can use the await with a foreach too
await foreach (var name in GetNamesAsync())
{
    // do some stuff
}
We also need to add async to our Main method to make it work
static async Task Main(string[] args)

Nullable Reference Types

As we all know there are two* types in C# : reference type and value type. struct is a typical example of a value type while class is a reference type. If you want more information about those two check this link.
In this section, we will talk about the famous, the one, the only null pointer exception and how to prevent it from happening.
It's too simple — you need just to activate the null reference types feature in your project and you will get a warning if you are doing something wrong.
// add this line to your .csproj
<NullableReferenceTypes>true</NullableReferenceTypes>
For example, if you are doing something like:
string name = null;
Console.WriteLine($"My name is {name}");
You will get a first warning saying "Assigning null to a non-nullable type."
We will fix it by changing string to string and checking if the name is null with ??.
string? name = null;
Console.WriteLine($"My name is {name ?? '?'}");
You can get his type of warning if you are Resharper too :p
* C# includes 3 types not 2 . I discovered the third one this week it's called "Pointer Type". I've never used this type but it looks like the C/C++ pointer.

No comments:

Post a Comment

No String Argument Constructor/Factory Method to Deserialize From String Value

  In this short article, we will cover in-depth the   JsonMappingException: no String-argument constructor/factory method to deserialize fro...