阅读VisualStudio2022公告和.NET6公告以了解更多信息,包括如何安装。
VisualStudio2022公告
.NET6
全局和隐式usings
using指令简化了你使用命名空间的方式。C#10包括一个新的全局using指令和隐式usings,以减少你需要在每个文件顶部指定的usings数量。
全局using指令
如果关键字global出现在using指令之前,则using适用于整个项目:
globalusingSystem;
你可以在全局using指令中使用using的任何功能。例如,添加静态导入类型并使该类型的成员和嵌套类型在整个项目中可用。如果你在using指令中使用别名,该别名也会影响你的整个项目:
globalusingstaticSystem.Console;globalusingEnv=System.Environment;
你可以将全局使用放在任何.cs文件中,包括Program.cs或专门命名的文件,如globalusings.cs。全局usings的范围是当前编译,一般对应当前项目。
有关详细信息,请参阅全局using指令。
隐式usings
隐式usings功能会自动为你正在构建的项目类型添加通用的全局using指令。要启用隐式usings,请在.csproj文件中设置ImplicitUsings属性:
enable
一些特定全局using指令集取决于你正在构建的应用程序的类型。例如,控制台应用程序或类库的隐式usings不同于ASP.NET应用程序的隐式usings。
Combiningusing功能
文件顶部的传统using指令、全局using指令和隐式using可以很好地协同工作。隐式using允许你在项目文件中包含适合你正在构建的项目类型的.NET命名空间。全局using指令允许你包含其他命名空间,以使它们在整个项目中可用。代码文件顶部的using指令允许你包含项目中仅少数文件使用的命名空间。
无论它们是如何定义的,额外的using指令都会增加名称解析中出现歧义的可能性。如果遇到这种情况,请考虑添加别名或减少要导入的命名空间的数量。例如,你可以将全局using指令替换为文件子集顶部的显式using指令。
如果你需要删除通过隐式usings包含的命名空间,你可以在项目文件中指定它们:
你还可以添加命名空间,就像它们是全局using指令一样,你可以将Using项添加到项目文件中,例如:
文件范围的命名空间
许多文件包含单个命名空间的代码。从C#10开始,你可以将命名空间作为语句包含在内,后跟分号且不带花括号:
namespaceMyCompany.MyNamespace;classMyClass//Note:noindentation
对lambda表达式和方法组的改进
微软对lambda的语法和类型进行了多项改进。微软预计这些将广泛有用,并且驱动方案之一是使ASP.NETMinimalAPI更加简单。
lambda的语法
ASP.NETMinimalAPI
lambda的自然类型
Lambda表达式现在有时具有“自然”类型。这意味着编译器通常可以推断出lambda表达式的类型。
到目前为止,必须将lambda表达式转换为委托或表达式类型。在大多数情况下,你会在BCL中使用重载的Func<...>或Action<...>委托类型之一:
Funcparse=(strings)=>int.Parse(s);
但是,从C#10开始,如果lambda没有这样的“目标类型”,微软将尝试为你计算一个:
varparse=(strings)=>int.Parse(s);
你可以在你最喜欢的编辑器中将鼠标悬停在varparse上,然后查看类型仍然是Func。一般来说,编译器将使用可用的Func或Action委托(如果存在合适的委托)。否则,它将合成一个委托类型(例如,当你有ref参数或有大量参数时)。
并非所有lambda表达式都有自然类型——有些只是没有足够的类型信息。例如,放弃参数类型将使编译器无法决定使用哪种委托类型:
varparse=s=>int.Parse(s);//ERROR:Notenoughtypeinfointhelambda
lambda的自然类型意味着它们可以分配给较弱的类型,例如object或Delegate:
objectparse=(strings)=>int.Parse(s);//FuncDelegateparse=(strings)=>int.Parse(s);//Func
当涉及到表达式树时,微软结合了“目标”和“自然”类型。如果目标类型是LambdaExpression或非泛型Expression(所有表达式树的基类型)并且lambda具有自然委托类型D,微软将改为生成Expression:
LambdaExpressionparseExpr=(strings)=>int.Parse(s);//Expression>ExpressionparseExpr=(strings)=>int.Parse(s);//Expression>
方法组的自然类型
方法组(即没有参数列表的方法名称)现在有时也具有自然类型。你始终能够将方法组转换为兼容的委托类型:
Funcread=Console.Read;Actionwrite=Console.Write;
现在,如果方法组只有一个重载,它将具有自然类型:
varread=Console.Read;//Justoneoverload;Funcinferredvarwrite=Console.Write;//ERROR:Multipleoverloads,can'tchoose
lambda的返回类型
在前面的示例中,lambda表达式的返回类型是显而易见的,并被推断出来的。情况并非总是如此:
varchoose=(boolb)=>b1:"two";//ERROR:Can'tinferreturntype
在C#10中,你可以在lambda表达式上指定显示返回类型,就像在方法或本地函数上一样。返回类型在参数之前。当你指定一个显示的返回类型时,参数必须用括号括起来,这样编译器或其他开发人员不会太混淆:
varchoose=object(boolb)=>b1:"two";//Func
lambda上的属性
从C#10开始,你可以将属性放在lambda表达式上,就像对方法和本地函数一样。当有属性时,lambda的参数列表必须用括号括起来:
Funcparse=[Example(1)](s)=>int.Parse(s);varchoose=[Example(2)][Example(3)]object(boolb)=>b1:"two";
就像本地函数一样,如果属性在AttributeTargets.Method上有效,则可以将属性应用于lambda。
Lambda的调用方式与方法和本地函数不同,因此在调用lambda时属性没有任何影响。但是,lambdas上的属性对于代码分析仍然有用,并且可以通过反射发现它们。
structs的改进
C#10为structs引入了功能,可在structs(结构)和类之间提供更好的奇偶性。这些新功能包括无参数构造函数、字段初始值设定项、记录结构和with表达式。
01无参数结构构造函数和字段初始值设定项
在C#10之前,每个结构都有一个隐式的公共无参数构造函数,该构造函数将结构的字段设置为默认值。在结构上创建无参数构造函数是错误的。
从C#10开始,你可以包含自己的无参数结构构造函数。如果你不提供,则将提供隐式无参数构造函数以将所有字段设置为默认值。你在结构中创建的无参数构造函数必须是公共的并且不能是部分的:
publicstructAddresspublicAddress()City="";publicstringCity{get;init;}
你可以如上所述在无参数构造函数中初始化字段,也可以通过字段或属性初始化程序初始化它们:
publicstructAddresspublicstringCity{get;init;}="";
通过默认创建或作为数组分配的一部分创建的结构会忽略显示无参数构造函数,并始终将结构成员设置为其默认值。有关结构中无参数构造函数的更多信息,请参阅结构类型。
02Recordstructs
从C#10开始,现在可以使用recordstruct定义record。这些类似于C#9中引入的record类:
publicrecordstructPersonpublicstringFirstName{get;init;}publicstringLastName{get;init;}
你可以继续使用record定义记录类,也可以使用record类来清楚地说明。
结构已经具有值相等——当你比较它们时,它是按值。记录结构添加IEquatable支持和==运算符。记录结构提供IEquatable的自定义实现以避免反射的性能问题,并且它们包括记录功能,如ToString()覆盖。
publicrecordstructPerson(stringFirstName,stringLastName);
要创建不可变的记录结构,请将readonly添加到结构(就像你可以添加到任何结构一样)或将readonly应用于单个属性。对象初始化器是可以设置只读属性的构造阶段的一部分。这只是使用不可变记录结构的一种方法:
varperson=newPerson{FirstName="Mads",LastName="Torgersen"};publicreadonlyrecordstructPersonpublicstringFirstName{get;init;}publicstringLastName{get;init;}
在本文中了解有关记录结构的更多信息。
记录结构
03Record类中ToString()上的密封修饰符
记录类也得到了改进。从C#10开始,ToString()方法可以包含seal修饰符,这会阻止编译器为任何派生记录合成ToString实现。
在本文中的记录中了解有关ToString()的更多信息。
有关ToString()的更多信息
04结构和匿名类型的表达式
C#10支持所有结构的with表达式,包括记录结构,以及匿名类型:
varperson2=personwith{LastName="Kristensen"};
这将返回一个具有新值的新实例。你可以更新任意数量的值。你未设置的值将保留与初始实例相同的值。
在本文中了解有关with的更多信息
了解有关with的更多信息
内插字符串改进
当微软在C#中添加内插字符串时,微软总觉得在性能和表现力方面,使用该语法可以做更多事情。
01内插字符串处理程序
今天,编译器将内插字符串转换为对string.Format的调用。这会导致很多分配——参数的装箱、参数数组的分配,当然还有结果字符串本身。此外,它在实际插值的含义上没有任何回旋余地。
在C#10中,微软添加了一个库模式,允许API“接管”对内插字符串参数表达式的处理。例如,考虑StringBuilder.Append:
varsb=newStringBuilder();sb.Append($"Hello{args[0]},howareyou");
到目前为止,这将使用新分配和计算的字符串调用Append(stringvalue)重载,将其附加到StringBuilder的一个块中。但是,Append现在有一个新的重载Append(refStringBuilder.AppendInterpolatedStringHandlerhandler),当使用内插字符串作为参数时,它优先于字符串重载。
通常,当你看到SomethingInterpolatedStringHandler形式的参数类型时,API作者在幕后做了一些工作,以更恰当地处理插值字符串以满足其目的。在微软的Append示例中,字符串“Hello”、args[0]和“,howareyou?”将单独附加到StringBuilder中,这样效率更高且结果相同。
有时你只想在特定条件下完成构建字符串的工作。一个例子是Debug.Assert:
Debug.Assert(condition,$"{SomethingExpensiveHensHere()}");
在大多数情况下,条件为真,第二个参数未使用。但是,每次调用都会计算所有参数,从而不必要地减慢执行速度。Debug.Assert现在有一个带有自定义插值字符串构建器的重载,它确保第二个参数甚至不被评估,除非条件为假。
最后,这是一个在给定调用中实际更改字符串插值行为的示例:String.Create()允许你指定IFormatProvider用于格式化插值字符串参数本身的洞中的表达式:
String.Create(CultureInfo.InvariantCulture,$"Theresultis{result}");
你可以在本文和有关创建自定义处理程序的本教程中了解有关内插字符串处理程序的更多信息。
创建自定义处理程序
内插字符串处理程序的更多信息
02常量内插字符串
如果内插字符串的所有洞都是常量字符串,那么生成的字符串现在也是常量。这使你可以在更多地方使用字符串插值语法,例如属性:
[Obsolete($"Call{nameof(Discard)}instead")]
请注意,必须用常量字符串填充洞。其他类型,如数字或日期值,不能使用,因为它们对文化敏感,并且不能在编译时计算。
其他改进
C#10对整个语言进行了许多较小的改进。其中一些只是使C#以你期望的方式工作。
intx2;inty2;(x2,y2)=(0,1);//WorksinC#9(varx,vary)=(0,1);//WorksinC#9(x2,vary3)=(0,1);//WorksinC#10onwards
改进的明确分配
如果你使用尚未明确分配的值,C#会产生错误。C#10可以更好地理解你的代码并且产生更少的虚假错误。这些相同的改进还意味着你将看到更少的针对空引用的虚假错误和警告。
扩展的属性模式
C#10添加了扩展属性模式,以便更轻松地访问模式中的嵌套属性值。例如,如果微软在上面的Person记录中添加一个地址,微软可以通过以下两种方式进行模式匹配:
objectobj=newPersonFirstName="Kathleen",LastName="Dollard",Address=newAddress{City="Seattle"}if(objisPerson{Address:{City:"Seattle"}})Console.WriteLine("Seattle");if(objisPerson{Address.City:"Seattle"})//ExtendedpropertypatternConsole.WriteLine("Seattle");
扩展属性模式简化了代码并使其更易于阅读,尤其是在匹配多个属性时。
调用者表达式属性
CallerArgumentExpressionAttribute提供有关方法调用上下文的信息。与其他CompilerServices属性一样,此属性应用于可选参数。在这种情况下,一个字符串:
voidCheckExpression(boolcondition,[CallerArgumentExpression("condition")]stringmessage=null)Console.WriteLine($"Condition:{message}");
传递给CallerArgumentExpression的参数名称是不同参数的名称。作为参数传递给该参数的表达式将包含在字符串中。例如,
vara=6;varb=true;CheckExpression(true);CheckExpression(b);CheckExpression(a>5);//Output://Condition:true//Condition:b//Condition:a>5
ArgumentNullException.ThrowIfNull()是如何使用此属性的一个很好的示例。它通过默认提供的值来避免必须传入参数名称:
voidMyMethod(objectvalue){ArgumentNullException.ThrowIfNull(value);}