C#6.0 在 2015 年7月随着.NET Framework 4.6 一同发布,后期发布了.NET Framework 4.6.1,4.6.2。
一、自动属性初始化(Auto-property initializers)
public class Account
{
public string Name { get; set; } = "summit";
public int Age { get; private set; } = 22;
public IList<int> AgeList { get; set; }= new List<int> { 10, 20, 30, 40, 50 };
}
二、字符串嵌入值(String interpolation)
在之前版本的String.Format中有多少个参数就要写多少个占位符还必须按照顺序,否则就报错。
新版本中在字符串前用$来标识后边的字符可以使用{对象}来作为占位符,而且支持智能提示。
tip:如果想输出{或}符号,写两个即可,如$"{{"。
Console.WriteLine($"年龄: {account.Age} 生日: {account.BirthDay.ToString("yyyy-MM-dd")} ");
string t2 = $"{account.Age}_ {account.BirthDay.ToString("yyyy-MM-dd")}";
Console.WriteLine($" {(account.Age<=22?"小鲜肉":"老鲜肉")} ");
三、导入静态方法(Using Static)
可用于导入单个类的静态方法。
using static System.Math;
还可以使用 using static
为具有静态和实例方法的类导入类的静态方法。
using static System.String;
在 LINQ 查询中可以通过导入 Enumerable 或 Queryable 来导入 LINQ 模式。
using static System.Linq.Enumerable;
四、空值条件运算符(Null-conditional operators)
检查变量是否为null,如果不为null那就执行。
老的语法,简单却繁琐。
public static Point FromJson(JObject json)
{
if (json != null &&
json["X"] != null &&
json["X"].Type == JTokenType.Integer &&
json["Y"] != null &&
json["Y"].Type == JTokenType.Integer
)
{
return new Point((int)json["X"], (int)json["Y"]);
}
return null;
}
?.和?[运算符化简成
public static Point FromJson(JObject json)
{
if (json?["X"]?.Type == JTokenType.Integer &&
json?["Y"]?.Type == JTokenType.Integer
)
{
return new Point((int)json["X"], (int)json["Y"]);
}
return null;
}
?.还有一个比较大的用处在触发事件的时候
if (onChanged != null)
{
onChanged(this, args);
}
现在可以改写成这样
OnChanged?.(this, args);
五、索引初始化集合(Index Initializers)
这种方式可以给字典或其他对象通过索引赋值初始化.
IDictionary<int, string> dict = new Dictionary<int, string>() {[1]="first", [2]="second" } ;
foreach(var dic in dict)
{
Console.WriteLine($"key: {dic.Key} value:{dic.Value}");
}
//output:
//key: 1 value:first
//key: 2 value:second
六、异常过滤器(Exception filters)
如果用于异常筛选器的表达式计算结果为 true
,则 catch 子句将对异常执行正常处理。 如果表达式计算结果为 false
,则将跳过 catch
子句。
try {
//异常抛出
}
catch (MyException ex) when ex.Status == MyExceptionStatus.A{
//do something
}
catch (MyException ex) when ex.Status== MyExceptionStatus.B{
//do something
}
catch (Exception caught) {
//do something
}
实例:
public static async Task<string> MakeRequest()
{
WebRequestHandler webRequestHandler = new WebRequestHandler();
webRequestHandler.AllowAutoRedirect = false;
using (HttpClient client = new HttpClient(webRequestHandler))
{
var stringTask = client.GetStringAsync("https://docs.microsoft.com/en-us/dotnet/about/");
try
{
var responseText = await stringTask;
return responseText;
}
catch (System.Net.Http.HttpRequestException e) when (e.Message.Contains("301"))
{
return "Site Moved";
}
}
}
七、nameof表达式 (nameof expressions)
nameof 表达式的计算结果为符号的名称。 每当需要变量、属性或成员字段的名称时,这是让工具正常运行的好办法。 nameof
的其中一个最常见的用途是提供引起异常的符号的名称。
在对方法参数进行检查时经常这样写:
private static void Add(Account account)
{
if (account == null)
throw new ArgumentNullException("account");
}
如果某天参数的名字被修改了,下面的字符串很容易漏掉忘记修改.
private static void Add(Account account)
{
if (account == null)
throw new ArgumentNullException(nameof(account));
}
八、在cath和finally语句块里使用await(Await in catch and finally blocks)
Resource res = null;
try
{
res = await Resource.OpenAsync(…); // You could do this.
…
}
catch(ResourceException e)
{
await Resource.LogAsync(res, e); // Now you can do this …
}
finally
{
if (res != null) await res.CloseAsync(); // … and this.
}
九、Lambda表达式在属性、方法中应用
1、在属性里使用Lambda表达式
注意属性是没有()的
public string Name => string.Format("姓名: {0}", "summit");
2、在方法成员上使用Lambda表达式
public void Print() => Console.WriteLine(Name);
static int LambdaFunc(int x, int y) => x*y;
原Point类代码较多:
public class Point
{
public int X { get; set; } public int Y { get; set; } public Point(int x, int y)
{
X = x;
Y = y;
} public double Dist
{
get { return Math.Sqrt(X * X + Y * Y); }
} public override string ToString()
{
return String.Format("({0}, {1})", X, Y);
}
}
运用属性初始化和方法Lambda简化后的Point类是这样的:
public class Point
{
public int X { get; } = 2;
public int Y { get; set; } = 1;
public double Dist => Sqrt(X * X + Y * Y);
public override string ToString() => $"({X}, {Y})";
}