15

C# 9: New `and`, `or`, and `not` Keywords for Pattern Matching

 4 years ago
source link: https://www.infoq.com/news/2020/07/CSharp-And-Or-Not/
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.
neoserver,ios ssh client

Though it may sound like our occasionalApril Fools Day joke, C# 9 is looking to add and , or , and not to its list of keywords. Specifically, for use in pattern matching.

In order to make pattern matching more flexible and powerful, C#’s designers want to add the concept of conjunctive, disjunctive, and negated patterns. Superficially, they look just like Boolean operations where you want to match on both patterns (conjunctive), either pattern (disjunctive) or not match on a pattern (negated).

And that’s the problem. If you are working with Booleans, the && and || operators would be ambiguous; the compiler wouldn’t be able to determine if they are referring to values or patterns. To illustrate this idea, consider this disjunctive pattern:

if (myBool is true or false)

This would be interpreted as “true if myBool equals true or if myBool equals false”.

If we use the && and || operators for combining patterns, then we get this syntax:

if (myBool is true || false)

But this statement already means “true if myBool equals the result of (true or false)”. Or to simplify it, “true if myBool equals true”. Which is a completely different result than what we’d get with above disjunctive pattern.

Hence the new and, or, and not keywords are necessary to avoid ambiguity. You can learn more about them in Champion "and, or, and not patterns" . In the C# 9 feature status they are marked as being merged into the master branch.

One question frequently raised is whether or not the same variable can be declared multiple times. For example,

if ((e1, e2) is (0, int x) or (int x, 0))
{
    M(x);
}

In this theoretical example, the variable x is either e1 or e2 . This would be the equivalent of writing,

case (0, int x):
case (int x, 0):
    M(x);

In fact, you would have to declare x in both patterns for x to be definitely assigned . If you used the below code instead, either x or y wouldn’t have a value.

if ((e1, e2) is (0, int x) or (int y, 0))
{
    M(x);
    M(y);
}

In order to this problem, the current proposal says,

beneath a not or or, pattern variables may not be declared.

This may be reconsidered in future versions of C#.


About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK