常见 C# 代码约定
编码约定对于在开发团队中维护代码可读性、一致性和协作至关重要。 遵循行业实践和既定准则的代码更易于理解、维护和扩展。 大多数项目通过代码约定强制要求样式一致。 dotnet/docs
和 dotnet/samples
项目并不例外。 在本系列文章中,你将了解我们的编码约定和用于强制实施这些约定的工具。 你可以按原样采用我们的约定,或修改它们以满足团队的需求。
我们对约定的选择基于以下目标:
- 正确性:我们的示例将会复制并粘贴到你的应用程序中。 我们希望如此,因此我们需要代码具有复原能力且正确无误,即使在多次编辑之后也是如此。
- 教学:示例的目的是教授 .NET 和 C# 的全部内容。 因此,我们不会对任何语言功能或 API 施加限制。 相反,这些示例会告知某个功能在何时会是良好的选择。
- 一致性:读者期望我们的内容提供一致的体验。 所有示例应遵循相同的样式。
- 采用:我们积极更新示例以使用新的语言功能。 这种做法提高了对新功能的认识,并且提高了所有 C# 开发人员对这些功能的熟悉程度。
重要事项
Microsoft 会使用这些准则来开发示例和文档。 它们摘自 .NET 运行时、C# 编码样式和 C# 编译器 (roslyn) 准则。 我们选择了这些准则,因为它们在几年的开放源代码开发中采用。 这些准则可帮助社区成员参与运行时和编译器项目。 它们是常见 C# 约定的一个示例,而不是权威列表(请参阅 框架设计准则 详细指南)。
教学和采用目标是文档编码约定不同于运行时和编译器约定的原因。 运行时和编译器对热路径具有严格的性能指标。 许多其他应用程序则并非如此。 我们的教学目标要求我们不会禁止任何构造。 相反,示例显示了何时应使用构造。 与大多数生产应用程序相比,我们在更新示例方面更加积极。 我们的采用目标要求我们显示你目前应该编写的代码,即使去年编写的代码无需更改。
本文将对我们的准则进行说明。 这些准则会随着时间推移而演变,你将发现不遵循我们的准则的示例。 我们欢迎推动这些示例合规的 PR,或促使我们关注应更新的示例的问题。 我们的准则是开放源代码的,因此我们欢迎 PR 和问题。 但如果你的提交将更改这些建议,请先提出一个问题以供讨论。 欢迎使用我们的准则,或根据你的需求对其进行调整。
工具和分析器
工具可帮助团队强制实施约定。 可以启用代码分析来强制实施你偏好的规则。 还可以创建 editorconfig,以便 Visual Studio 可自动强制实施样式准则。 首先,可以复制 dotnet/docs
.editorconfig 以使用我们的样式。
借助这些工具,团队可以更轻松地采用首选的准则。 Visual Studio 在所有 .编辑器配置 范围内的文件来格式化你的代码。 可以使用多个配置来强制实施企业范围的约定、团队约定,甚至精细的项目约定。
代码分析在检测到规则冲突时生成警告和诊断。 可以配置想要应用于项目的规则。 然后,每个 CI 生成会在违反任何规则时通知开发人员。
诊断 ID
- 生成自己的分析器时选择适当的诊断 ID
语言准则
以下部分介绍了 .NET 文档团队在准备代码示例和示例时遵循的做法。 一般情况下,请遵循以下做法:
- 尽可能利用新式语言功能和 C# 版本。
- 避免过时的语言构造。
- 只捕获可以正确处理的异常;避免捕获一般异常。 例如,在没有异常筛选器的情况下,示例代码不应捕获 System.Exception 类型。
- 使用特定的异常类型提供有意义的错误消息。
- 使用 LINQ 查询和方法进行集合操作,以提高代码可读性。
- 将异步编程与异步和等待用于 I/O 绑定操作。
- 请谨慎处理死锁,并在适当时使用 Task.ConfigureAwait。
- 对数据类型而不是运行时类型使用语言关键字。 例如,使用
string
而不是 System.String,或使用int
而不是 System.Int32。 此建议包括使用类型nint
和nuint
。 - 使用
int
而不是无符号类型。int
的使用在整个 C# 中很常见,并且当你使用int
时,更易于与其他库交互。 特定于无符号数据类型的文档例外。。 - 仅当读者可以从表达式推断类型时使用
var
。 读者可在文档平台上查看我们的示例。 它们没有悬停或显示变量类型的工具提示。 - 以简洁明晰的方式编写代码。
- 避免过于复杂和费解的代码逻辑。
遵循更具体的准则。
字符串数据
使用字符串内插来连接短字符串,如下面的代码所示。
string displayName = $"{nameList[n].LastName}, {nameList[n].FirstName}";
若要在循环中追加字符串,尤其是在使用大量文本时,请使用 System.Text.StringBuilder 对象。
var phrase = "lalalalalalalalalalalalalalalalalalalalalalalalalalalalalala"; var manyPhrases = new StringBuilder(); for (var i = 0; i < 10000; i++) { manyPhrases.Append(phrase); } //Console.WriteLine("tra" + manyPhrases);
优先选择原始字符串字面量是转义序列或逐字字符串。
var message = """ This is a long message that spans across multiple lines. It uses raw string literals. This means we can also include characters like \n and \t without escaping them. """;
使用基于表达式的字符串内插,而不是位置字符串内插。
// Execute the queries. Console.WriteLine("scoreQuery:"); foreach (var student in scoreQuery) { Console.WriteLine($"{student.Last} Score: {student.score}"); }
构造函数和初始化
对记录类型的主构造函数参数使用 Pascal 例:
public record Person(string FirstName, string LastName);
在类和结构体类型的主构造函数参数中使用驼峰形。
使用
required
属性而不是构造函数来强制初始化属性值:public class LabelledContainer<T>(string label) { public string Label { get; } = label; public required T Contents { get; init; } }
数组和集合
- 使用集合表达式初始化所有集合类型:
string[] vowels = [ "a", "e", "i", "o", "u" ];
委托
- 使用
Func<>
和Action<>
,而不是定义委托类型。 在类中,定义委托方法。
Action<string> actionExample1 = x => Console.WriteLine($"x is: {x}");
Action<string, string> actionExample2 = (x, y) =>
Console.WriteLine($"x is: {x}, y is {y}");
Func<string, int> funcExample1 = x => Convert.ToInt32(x);
Func<int, int, int> funcExample2 = (x, y) => x + y;
- 使用
Func<>
或Action<>
委托定义的签名来调用方法。
actionExample1("string for x");
actionExample2("string for x", "string for y");
Console.WriteLine($"The value is {funcExample1("1")}");
Console.WriteLine($"The sum is {funcExample2(1, 2)}");
如果创建委托类型的实例,请使用简洁的语法。 在类中,定义委托类型和具有匹配签名的方法。
public delegate void Del(string message); public static void DelMethod(string str) { Console.WriteLine("DelMethod argument: {0}", str); }
创建委托类型的实例,然后调用该实例。 以下声明显示了紧缩的语法。
Del exampleDel2 = DelMethod; exampleDel2("Hey");
以下声明使用了完整的语法。
Del exampleDel1 = new Del(DelMethod); exampleDel1("Hey");
try-catch
和 using
语句正在异常处理中
对大多数异常处理使用 try-catch 语句。
static double ComputeDistance(double x1, double y1, double x2, double y2) { try { return Math.Sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2)); } catch (System.ArithmeticException ex) { Console.WriteLine($"Arithmetic overflow or underflow: {ex}"); throw; } }
通过使用 C# using 语句简化你的代码。 如果具有 try-finally 语句(该语句中
finally
块的唯一代码是对 Dispose 方法的调用),请使用using
语句代替。在以下示例中,
try-finally
语句仅在Dispose
块中调用finally
。Font bodyStyle = new Font("Arial", 10.0f); try { byte charset = bodyStyle.GdiCharSet; } finally { bodyStyle?.Dispose(); }
可以使用
using
语句执行相同的操作。using (Font arial = new Font("Arial", 10.0f)) { byte charset2 = arial.GdiCharSet; }
使用不需要大括号的新
using
语法:using Font normalStyle = new Font("Arial", 10.0f); byte charset3 = normalStyle.GdiCharSet;
&&
和 ||
运算符
在执行比较时,使用
&&
而不是&
,使用||
而不是|
,如以下示例所示。Console.Write("Enter a dividend: "); int dividend = Convert.ToInt32(Console.ReadLine()); Console.Write("Enter a divisor: "); int divisor = Convert.ToInt32(Console.ReadLine()); if ((divisor != 0) && (dividend / divisor) is var result) { Console.WriteLine("Quotient: {0}", result); } else { Console.WriteLine("Attempted division by 0 ends up here."); }
如果除数为 0,则 if
语句中的第二个子句将导致运行时错误。 但是,当第一个表达式为 false 时,&& 运算符将发生短路。 也就是说,它并不评估第二个表达式。 如果 divisor
为 0,则 & 运算符将同时计算这两个表达式,从而导致运行时错误。
new
运算符
当变量类型与对象类型匹配时,请使用对象实例化的简明形式之一,如以下声明所示。 当变量是接口类型或运行时类型的基类时,此窗体无效。
var firstExample = new ExampleClass();
ExampleClass instance2 = new();
前面的声明等效于下面的声明。
ExampleClass secondExample = new ExampleClass();
使用对象初始值设定项简化对象创建,如以下示例中所示。
var thirdExample = new ExampleClass { Name = "Desktop", ID = 37414, Location = "Redmond", Age = 2.3 };
下面的示例设置了与前面的示例相同的属性,但未使用初始值设定项。
var fourthExample = new ExampleClass(); fourthExample.Name = "Desktop"; fourthExample.ID = 37414; fourthExample.Location = "Redmond"; fourthExample.Age = 2.3;
事件处理
- 使用 lambda 表达式定义稍后无需移除的事件处理程序:
public Form2()
{
this.Click += (s, e) =>
{
MessageBox.Show(
((MouseEventArgs)e).Location.ToString());
};
}
Lambda 表达式缩短了以下传统定义。
public Form1()
{
this.Click += new EventHandler(Form1_Click);
}
void Form1_Click(object? sender, EventArgs e)
{
MessageBox.Show(((MouseEventArgs)e).Location.ToString());
}
静态成员
使用类名调用 static 成员:ClassName.StaticMember。 这种做法通过明确静态访问使代码更易于阅读。 请勿使用派生类的名称来限定基类中定义的静态成员。 虽然该代码编译,但代码可读性具有误导性,如果向派生类添加具有相同名称的静态成员,则代码将来可能会中断。
LINQ 查询
对查询变量使用有意义的名称。 下面的示例为位于西雅图的客户使用
seattleCustomers
。var seattleCustomers = from customer in Customers where customer.City == "Seattle" select customer.Name;
使用别名确保匿名类型的属性名称都使用 Pascal 大小写格式正确大写。
var localDistributors = from customer in Customers join distributor in Distributors on customer.City equals distributor.City select new { Customer = customer, Distributor = distributor };
如果结果中的属性名称模棱两可,请对属性重命名。 例如,如果查询返回客户名称和分发服务器名称,而不是将它们保留为结果中的
Name
形式,请重命名它们以澄清CustomerName
是客户的名称,DistributorName
是分发服务器的名称。var localDistributors2 = from customer in Customers join distributor in Distributors on customer.City equals distributor.City select new { CustomerName = customer.Name, DistributorName = distributor.Name };
在查询变量和范围变量的声明中使用隐式类型化。 有关 LINQ 查询中隐式类型的本指导会替代适用于隐式类型本地变量的一般规则。 LINQ 查询通常使用创建匿名类型的投影。 其他查询表达式使用嵌套泛型类型创建结果。 隐式类型变量通常更具可读性。
var seattleCustomers = from customer in Customers where customer.City == "Seattle" select customer.Name;
对齐
from
子句下的查询子句,如上面的示例所示。在其他查询子句前面使用
where
子句,确保后面的查询子句作用于经过缩减和筛选的一组数据。var seattleCustomers2 = from customer in Customers where customer.City == "Seattle" orderby customer.Name select customer;
使用多个
from
条款 而不是join
条款。 例如,Student
对象的集合可能包含测验分数的集合。 执行以下查询时,它将返回超过 90 的每个分数,以及收到分数的学生的家庭名称。var scoreQuery = from student in students from score in student.Scores where score > 90 select new { Last = student.LastName, score };
隐式类型本地变量
当变量的类型在赋值右侧比较明显时,对局部变量使用隐式类型。
var message = "This is clearly a string."; var currentTemperature = 27;
当类型在赋值右侧不明显时,请勿使用 var。 请勿假设类型明显来自方法名称。 如果变量类型是一个
new
运算符、显式转置或对字面值赋值,变量类型就被认为是清晰的。int numberOfIterations = Convert.ToInt32(Console.ReadLine()); int currentMaximum = ExampleClass.ResultSoFar();
不要使用变量名称指定变量的类型。 它可能不正确。 请改用类型来指定类型,并使用变量名称来指示变量的语义信息。 以下示例应对类型使用
string
,并使用类似iterations
的内容指示从控制台读取的信息的含义。var inputInt = Console.ReadLine(); Console.WriteLine(inputInt);
避免使用
var
来代替 dynamic。 如果想要进行运行时类型推理,请使用dynamic
。 有关详细信息,请参阅使用类型 dynamic(C# 编程指南)。在
for
循环中对循环变量使用隐式类型。下面的示例在
for
语句中使用隐式类型化。var phrase = "lalalalalalalalalalalalalalalalalalalalalalalalalalalalalala"; var manyPhrases = new StringBuilder(); for (var i = 0; i < 10000; i++) { manyPhrases.Append(phrase); } //Console.WriteLine("tra" + manyPhrases);
不要使用隐式类型化来确定
foreach
循环中循环变量的类型。 在大多数情况下,集合中的元素类型并不明显。 不应仅依靠集合的名称来推断其元素的类型。下面的示例在
foreach
语句中使用显式类型化。foreach (char ch in laugh) { if (ch == 'h') { Console.Write("H"); } else { Console.Write(ch); } } Console.WriteLine();
对 LINQ 查询中的结果序列使用隐式类型。 关于 LINQ 的部分说明了许多 LINQ 查询会导致必须使用隐式类型的匿名类型。 其他查询则会产生嵌套泛型类型,其中
var
的可读性更高。说明
注意不要意外更改可迭代集合的元素类型。 例如,可以轻松地在
foreach
语句中从 System.Linq.IQueryable 切换到 System.Collections.IEnumerable,这会更改查询的执行。
我们的一些示例解释了表达式的自然类型。 这些示例必须使用 var
,以便编译器选取自然类型。 即使这些示例不太明显,但示例必须使用 var
。 文本应解释该行为。
文件作用域命名空间声明
大多数代码文件声明单个命名空间。 因此,我们的示例应使用文件范围的命名空间声明:
namespace MySampleCode;
将 using 指令放在命名空间声明之外
当 using
指令位于命名空间声明之外时,该导入的命名空间是其完全限定的名称。 完全限定的名称更加清晰。 如果 using
指令位于命名空间内部,则它可以是相对于该命名空间的,也可以是它的完全限定名称。
using Azure;
namespace CoolStuff.AwesomeFeature
{
public class Awesome
{
public void Stuff()
{
WaitUntil wait = WaitUntil.Completed;
// ...
}
}
}
假设存在对 WaitUntil 类的引用(直接或间接)。
现在,让我们稍作改动:
namespace CoolStuff.AwesomeFeature
{
using Azure;
public class Awesome
{
public void Stuff()
{
WaitUntil wait = WaitUntil.Completed;
// ...
}
}
}
今天的编译成功了。 明天的也没问题。 但在下周的某个时候,前面(未改动)的代码失败,并出现两个错误:
- error CS0246: The type or namespace name 'WaitUntil' could not be found (are you missing a using directive or an assembly reference?)
- error CS0103: The name 'WaitUntil' does not exist in the current context
在命名空间中引入该类的其中一个依赖关系以 .Azure
:
namespace CoolStuff.Azure
{
public class SecretsManagement
{
public string FetchFromKeyVault(string vaultId, string secretId) { return null; }
}
}
放置在命名空间中的 using
指令与上下文相关,使名称解析复杂化。 在此示例中,它是它找到的第一个命名空间。
CoolStuff.AwesomeFeature.Azure
CoolStuff.Azure
Azure
添加匹配 CoolStuff.Azure
或 CoolStuff.AwesomeFeature.Azure
的新命名空间将在全局 Azure
命名空间前匹配。 可以通过向 global::
声明添加 using
修饰符来解决此问题。 但是,改为将 using
声明放在命名空间之外更容易。
namespace CoolStuff.AwesomeFeature
{
using global::Azure;
public class Awesome
{
public void Stuff()
{
WaitUntil wait = WaitUntil.Completed;
// ...
}
}
}
样式指南
一般情况下,对代码示例使用以下格式:
- 使用四个空格缩进。 不要使用选项卡。
- 一致地对齐代码以提高可读性。
- 将行限制为 65 个字符,以增强文档上的代码可读性,尤其是在移动屏幕上。
- 通过将长语句分解为多行来提高清晰度和用户体验。
- 对大括号使用“Allman”样式:左和右大括号另起一行。 大括号与当前缩进级别对齐。
- 如有必要,应在二进制运算符之前换行。
注释样式
使用单行注释(
//
)以进行简要说明。避免使用多行注释(
/* */
)来进行较长的解释。
代码示例中的注释未本地化。 这意味着代码中嵌入的解释不会被翻译。 较长的解释性文本应放在配套文章中,以便对其进行本地化。若要描述方法、类、字段和所有公共成员,请使用 XML 注释。
将注释放在单独的行上,而非代码行的末尾。
以大写字母开始注释文本。
以句点结束注释文本。
在注释分隔符 (
//
) 与注释文本之间插入一个空格,如下面的示例所示。// The following declaration creates a query. It does not run // the query.
布局约定
好的布局利用格式设置来强调代码的结构并使代码更便于阅读。 Microsoft 示例和样本符合以下约定:
使用默认的代码编辑器设置(智能缩进、4 字符缩进、制表符保存为空格)。 有关详细信息,请参阅选项、文本编辑器、C#、格式设置。
每行只写一条语句。
每行只写一个声明。
如果连续行未自动缩进,请将它们缩进一个制表符位(四个空格)。
在方法定义与属性定义之间添加至少一个空白行。
使用括号突出表达式中的子句,如下面的代码所示。
if ((startX > endX) && (startX > previousX)) { // Take appropriate action. }
例外情况出现在示例解释运算符或表达式优先级时。
安全性
请遵循安全编码准则中的准则。