Jump to content
Xtreme .Net Talk

Calling methods is easier and faster with C# 13 params collections


Recommended Posts

Guest Kathleen Dollard
Posted

C# 13 brings features that make it easier, safer, and faster to write code in the styles you know and love. You’ll find a full list of C# 13 features at What’s new in C# 13.

 

C# 13 fulfills a long standing feature request by allowing [iCODE]params[/iCODE] to be any of the collections supported by collection expressions, rather than just arrays. This feature builds on collection expressions that were introduced in C# 12.

 

[HEADING=1]Laying the groundwork with collection expressions[/HEADING]

 

Collection expressions were introduced in C# 12 and offer an alternative to the myriad of ways that you previously created different kinds of collections. In the context of the method, this allows you to simplify use of collections in many scenarios, including calling methods:

 

 

// Prior to C# 12WriteByteArray(new[] { (byte)1, (byte)2, (byte)3 });WriteByteSpan(stackalloc[] { (byte)1, (byte)2, (byte)3 });// After C# 12WriteByteArray([1, 2, 3]);WriteByteSpan([1, 2, 3]);static void WriteByteArray(byte[] bytes) { }static void WriteByteSpan(Span<byte> bytes) { }

 

 

This also allowed us to unify aspects of how C# works with collections. Note that collection expressions infer both the type of the collection and the type of the members.

 

[HEADING=1][iCODE]params[/iCODE] arrays[/HEADING]

 

[iCODE]params[/iCODE] has been in the language since C# 1.0. It allows calling code to include zero to many arguments as a comma delimited list. The method receives this list as an array, which might be empty:

 

 

WriteByteArray(1, 2, 3);WriteByteArray();static void WriteByteArray(params byte[] bytes) { }

 

 

Prior to C# 13, the [iCODE]params[/iCODE] must be declared as an array in the method’s parameter list.

 

In addition to calling with a list of values, you have always been able to call methods that take a [iCODE]params[/iCODE] with an array, and starting in C# 12, this can be a collection expression:

 

 

WriteByteArray(1, 2, 3);WriteByteArray([1, 2, 3]);byte[] bytes = [4, 5];WriteByteArray([1, 2, 3, .. bytes]);static void WriteByteArray(params byte[] bytes) { }

 

[HEADING=1][iCODE]params[/iCODE] collections[/HEADING]

 

Starting in C# 13, [iCODE]params[/iCODE] can be any of the collection types that support collection expressions:

 

 

WriteByteSpan(1, 2, 3);WriteByteSpan();static void WriteByteSpan(params Span<byte> bytes) { }

 

 

While that seems like a small change, it often enables the compiler to optimize your code. For example, if the method uses [iCODE]params Span<T>[/iCODE], the compiler can use stack space to create the span for the method. That performs better than allocating an array.

 

Using a specific type can also communicate to callers how the collection will be used. For example, [iCODE]params IReadonlyList<T>[/iCODE] indicates that the collection will not be changed.

 

If you use [iCODE]params IEnumerable<T>[/iCODE], your users can pass a list of literal values, an array, a [iCODE]List<T>[/iCODE], any of the collection types that implement [iCODE]IEnumerable<T>[/iCODE], or a LINQ expression:

 

 

WriteByteArray(1, 2, 3);byte[] bytes = [1, 2, 3, 4, 5];WriteByteArray(bytes.Where(x => x < 4));static void WriteByteArray(params IEnumerable<byte> bytes) { }

 

 

When the [iCODE]params[/iCODE] receiver is an interface and the argument is a discrete list of elements or a collection expression, the compiler will use a concrete type. Many collection interfaces have a single implementation that would be a logical choice, such as [iCODE]ReadonlyCollection<T>[/iCODE] for [iCODE]IReadonlyList<T>[/iCODE]. Since [iCODE]IEnumerable<T>[/iCODE] does not have such an obvious choice and is so fundamental to .NET, it uses a special high-performance type for both [iCODE]param[/iCODE] and collection expressions.

 

[HEADING=1]Overloading[/HEADING]

 

C# supports overloading methods – which means that multiple methods with the same name can exist, if they differ by the types of the parameters. You can overload on a [iCODE]params[/iCODE] collection, as you would other parameters.

 

For example, you might have an overload with [iCODE]params IEnumerable<T>[/iCODE] and one with [iCODE]params ReadOnlySpan<T>[/iCODE] in the same resolution scope. If you pass a list of values, the [iCODE]params ReadOnlySpan<T>[/iCODE] overload will be selected. If you pass an array, the [iCODE]params ReadOnlySpan<T>[/iCODE] overload will also be selected, because there is an implicit conversion from array to [iCODE]ReadONlySpan<T>[/iCODE]. If you pass a [iCODE]List<T>[/iCODE], the [iCODE]params IEnumerable<T>[/iCODE] overload will be used:

 

 

public class ParamCollections{   public static void Overloads()   {       WriteNumbers(1, 2, 3);       WriteNumbers([1, 2, 3]);       WriteNumbers(new[] { 1, 2, 3 });       byte[] ints = [1, 2, 3, 4, 5];       WriteNumbers(ints.Where(x => x < 4));   }   // This code shifts from static local functions to private functions    // because overloading is not allowed for local functions.       private static void WriteNumbers<T>(params IEnumerable<T> values) => Console.WriteLine("IEnumerable");   private static void WriteNumbers<T>(params ReadOnlySpan<T> values) => Console.WriteLine("Span");}// The result is:// // Span// Span// Span// IEnumerable

 

 

The compiler picks a reasonable overload for you. It will often be [iCODE]Span<T>[/iCODE] if it is available because this saves an allocation in the method call.

 

[iCODE]params[/iCODE] collections will make your applications faster, whether or not you add them to your own code, because the .NET runtime libraries can now use high performance types like [iCODE]Span<T>[/iCODE] in more places. You use the same concepts you’ve always used for [iCODE]params[/iCODE], callers have more flexibility in how they call the method, and the compiler picks the overload.

 

[HEADING=1]Considering overloads[/HEADING]

 

Overloads are a very powerful feature of C#. But as with many powerful things, proper usage is important. If there are multiple methods with the same name, they should do the same thing. They may differ in how they do it or differ by performance, but changing the types passed should not introduce breaking changes in your application. Often, this means calling a single implementation after adjusting the argument values.

 

As you can see in the example above, small changes in the calling code can result in different overloads being called. This is true for all C# applications, whether or not you use [iCODE]params[/iCODE] collections.

 

[HEADING=1]Summary[/HEADING]

 

We’re excited about [iCODE]params[/iCODE] collection in C# 13 and can’t wait to hear what you think!

 

You can learn about all the features in C# 13 at What’s new in C# 13.

 

The post Calling methods is easier and faster with C# 13 params collections appeared first on .NET Blog.

 

Continue reading...

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.

Guest
Reply to this topic...

×   Pasted as rich text.   Paste as plain text instead

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

×
×
  • Create New...