CSharp 6.0 (C# 6.0)
Eintrag zuletzt aktualisiert am: 19.10.2015
C# 6.0 (alias C# 2015) ist der Nachfolger von
C# 5.0 und ab
.NET Framework 4.6 und
Visual Studio 2015 verfügbar.
C# 6 ist (zusammen mit Visual Basic 14 alias
Visual Basic 2015) am 20.7.2015 im Rahmen von
.NET Framework 4.6 und
Visual Studio 2015 erschienen.
Hinweis: einige der neuen Sprachfeatures funktionieren auch, wenn man mit
Visual Studio 2015 für ältere .NET Framework-Versionen kompiliert, da diese Features reine
Compiler-Features sind, die keine der neuen Klassen in
.NET 4.6 erfordern und der entstehende IL-Code kompatibel zu früheren .NET-Versionen ist.
Geschichte
Erste Ankündigungen gab es auf der NDC 2013-Konferenz.
Eine breitere Vorstellung gab es auf der
BUILD 2014-Konferenz Anfang April 2014.
C# 6.0 ist am 20.7.2015 erschienen.
Neue Sprachfeatures in C# 6.0
Auto-property initializers
public int X { get; set; } = x;
Getter-only auto-properties
public int Y { get; } = y;
Using static members
using
System.Console; … WriteLine(4);
Dictionary initializer
new JObject { ["x"] = 3, ["y"] = 7 }
Await in catch/finally
try … catch {
await … } finally {
await … }
Exception filters
catch(E e) if (e.Count > 5) { … }
Expression-bodied members
public double Dist => Sqrt(X * X + Y * Y);
Null propagation Operator
customer?.Orders?[5]?.$price
Details zu den neuen Sprachfeatures
Zu den sehr praktischen Neuerungen in beiden Sprachen gehört der Fragezeichen-Punkt-Operator (?.), der im Gegensatz zu dem einfachen Punkt-Operator keinen Laufzeitfehler auslöst, wenn der Ausdruck vor dem Punkt keinen Wert besitzt, also "null" (in C#) beziehungsweise "nothing" (in
Visual Basic .NET) liefert.
string name = repository?.GetKontakt()?.Name;
Mit einer
String Interpolation können Entwickler die Zusammensetzung von Zeichenketten aus festen und variablen Bestandteilen übersichtlicher als bisher realisieren.
var ausgabeAlt = String.Format("Kontakt #{0:0000}: {1} ist in der
Liste seit {2:d}.", k.ID, k.GanzerName, k.ErzeugtAm);
var ausgabeNeu = $"Kontakt #{k.ID:0000}: {k.GanzerName} ist in der
Liste seit {k.ErzeugtAm:d}.";
Zudem kann man nun automatische Properties, für die es keine explizite Felddeklaration gibt, direkt im Rahmen der Deklaration mit einem Wert initialisieren und auch automatische Properties schaffen, die nach ihrer Initialisierung unveränderbar sind. Lediglich im Konstruktor der Klasse kann der Entwickler solche Eigenschaften dann noch ändern (siehe Listing 1).
public class Kontakt
{
// Automatic Properties mit Initialisierung
public string Land { get; set; } = "Deutschland";
// Automatic Properties mit Initialisierung und ohne
Setter
public DateTime ErzeugtAm { get; } = DateTime.Now;
public Kontakt(DateTime erzeugtAm)
{
//
Getter Only Auto
Property im Konstruktor setzen
ErzeugtAm = DateTime.Now;
}
}
Listing 1: Automatische Properties mit Initialisierung und optional auch ohne
Setter
Der neue Operator nameof() liefert den Namen eines Bezeichners als Zeichenkette (bei mehrgliederigen Namen nur den letzten Teil). Dieser Operator erhöht die Robustheit und erleichtert das
Refactoring in Situationen, in denen der Name einer Klasse oder eines Klassenmitglieds als Zeichenkette zu übergeben ist, z.B. ArgumentNullException (siehe Listing 2) Dependency Properties und
PropertyChangedEventArgs (siehe Listing 3).
public void SaveKontakt(Kontakt neuerKontakt)
{
if (neuerKontakt == null) throw new ArgumentNullException(nameof(neuerKontakt));
...
}
Listing 2: Einsatz des Operators nameof() für ArgumentNullException
public int KontaktAnzahl
{
get { return kontaktAnzahl; }
set
{
PropertyChanged(this, new
PropertyChangedEventArgs(nameof(KontaktAnzahl)));
kontaktAnzahl = value;
}
}
Listing 3: Einsatz des Operators nameof() für
PropertyChangedEventArgs
Alle vorgenannten Sprachfeatures hat Microsoft sowohl in C# 6 als auch Visual Basic 14 ergänzt – auch wenn sie hier aus Platzgründen nur in C# abgedruckt sind. Andere Sprachfeatures wurden in lediglich einer Sprache ergänzt, weil es sie in der anderen bereits gab.
Dazu gehören
Exception Filter, mit denen der C#-Entwickler nun zusätzlich zu den Exception-Klassen in den catch-Blöcken mit dem Schlüsselwort when zwischen verschiedenen Fällen differenzieren kann (siehe Listing 4). Diese Spracheigenschaft gibt es in
Visual Basic .NET schon seit dem Jahr 2002. Ebenso hat C# nun bei der Möglichkeit gleichgezogen, statische Klassen mit using so einzubinden, dass man auf die einzelnen Klassenmitglieder nun ohne Verwendung des Klassennamens zugreifen darf:
// bisherige Schreibweise
Console.WriteLine(Environment.UserDomainName + @"\" + Environment.UserName);
// neu in C# 6.0
using static
System.Console;
using static System.Environment;
…
WriteLine(UserDomainName + @"\" + UserName);
Dieses Sprachfeature ist jedoch umstritten, weil hier die Lesbarkeit des Programmcodes zugunsten einer ersparten Tipparbeit geopfert wird.
try
{
var datei =
System.IO.File.ReadLines(filename);
}
catch (ArgumentException) when (filename == "")
{
Console.WriteLine("Ohne Dateiname macht diese Aktion keinen Sinn!");
}
catch (ArgumentException ex) when (ex.Message.Contains("Illegales"))
{
Console.WriteLine("Ungültige Zeichen im Dateinamen: " + filename);
}
catch (ArgumentException ex)
{
Console.WriteLine("Ungültige Angabe: " + filename + ":" + ex.Message);
}
catch (NotSupportedException ex) when (ex.Message.Contains("format"))
{
Console.WriteLine("Ungültiges Format!");
}
catch (NotSupportedException ex)
{
Console.WriteLine("Nicht unterstützt: " + ex.Message);
}
catch (FileNotFoundException ex)
{
Console.WriteLine("Datei " + filename + " nicht gefunden");
}
catch (Exception ex)
{
Console.WriteLine("Anderer Fehler: " + ex.Message);
}
Listing 4:
Exception Filter in C# 6
Auch Visual Basic hat ein paar Angleichungen an C# erfahren: Erlaubt sind nun mehrzeilige Zeichenkettenliterale und Kommentare in
LINQ-Ausdrücken sowie partielle Module und partielle Schnittstellendefinitionen.
C# hat aber in der Version 6 auch Möglichkeiten bekommen, die es in
Visual Basic .NET nicht gab und auch vorerst weiterhin nicht geben wird. Dazu gehören sogenannte Expression-bodied Members:
Methoden und nicht beschreibbare Properties, die nur einen einzigen Ausdruck zurückliefern, kann der C#-Entwickler nun verkürzt unter Einsatz des Lambda-Operators => schreiben:
public string GanzerName => this.Vorname + " " + this.Nachname;
public decimal NeuerEinkauf(decimal wert) => this.Umsatz += wert;
public override string ToString() => this.GanzerName + ": " + this.KontaktStatus;
Weiterhin darf ein C#-Entwickler nun die Schlüsselwörter async und
await auch in catch- und finally-Blöcken verwenden. Auch dies ist bisher für
Visual Basic .NET nicht vorgesehen, sodass die komplette Sprachparität zwischen den beiden vorherrschenden
.NET-Sprachen nicht gegeben ist.
Leider wieder gestrichene (also doch nicht in C# 6.0 enthaltene) Sprachfeatures
Declaration expressions
int.TryParse(s, out var x);
Params IEnumerabl
e int Avg(params
IEnumerable<int> numbers) { … }
Primary constructors
class Point(int x, int y) { … }
Binary literals
0b00000100
Digit separators
0xEF
FF_00A0
Event initializers
new Customer { Notify += My
Handler };
Field targets on autoprops
<
Field: Serializable>
Property p As Integer