Общие соглашения о коде C#
Соглашения по программированию необходимы для поддержания удобочитаемости кода, согласованности и совместной работы в команде разработчиков. Код, который соответствует отраслевым методикам и установленным рекомендациям, проще понимать, поддерживать и расширять. Большинство проектов применяют согласованный стиль с помощью соглашений о коде. И dotnet/docs
dotnet/samples
проекты не являются исключением. В этой серии статей вы узнаете о наших соглашениях по программированию и средствах, которые мы используем для их применения. Вы можете принять наши соглашения как есть или изменить их в соответствии с потребностями вашей команды.
Мы выбрали наши соглашения на основе следующих целей:
- Правильность. Наши примеры копируются и вставляются в приложения. Мы ожидаем, что поэтому нам нужно сделать код устойчивым и правильным, даже после нескольких изменений.
- Обучение. Цель наших примеров — научить все .NET и C#. По этой причине мы не размещаем ограничения на какие-либо языковые функции или API. Вместо этого эти примеры учат, когда функция является хорошим выбором.
- Согласованность. Читатели ожидают согласованный интерфейс в нашем содержимом. Все образцы должны соответствовать одному и тому же стилю.
- Внедрение: мы агрессивно обновляем наши примеры, чтобы использовать новые языковые функции. Эта практика повышает осведомленность о новых функциях и делает их более знакомыми для всех разработчиков C#.
Внимание
Эти рекомендации используются корпорацией Майкрософт для разработки примеров и документации. Они были приняты из правил среды выполнения .NET, стиля написания кода C# и компилятора C# (roslyn). Мы выбрали эти рекомендации из-за их внедрения в течение нескольких лет разработки с открытым кодом. Эти рекомендации помогают членам сообщества участвовать в проектах среды выполнения и компилятора. Они предназначены быть примером общепринятых соглашений C#, а не авторитетным списком (см. «Руководства по проектированию фреймворков» для получения подробных рекомендаций).
Цели обучения и внедрения определяют, почему соглашение о кодировании документов отличается от соглашений среды выполнения и компилятора. Среда выполнения и компилятор имеют строгие метрики производительности для горячих путей. Многие другие приложения не делают. Наша цель обучения требует, чтобы мы не запрещали какую-либо конструкцию. Вместо этого примеры показывают, когда должны использоваться конструкции. Мы обновляем примеры более агрессивно, чем большинство рабочих приложений. Наша цель внедрения требует, чтобы мы отображали код, который вы должны написать сегодня, даже если код, написанный в прошлом году, не нуждается в изменениях.
В этой статье описываются наши рекомендации. Рекомендации развиваются со временем, и вы найдете примеры, которые не соответствуют нашим рекомендациям. Мы приветствуем PR, которые приносят эти образцы в соответствие или проблемы, которые обращают внимание на примеры, которые мы должны обновить. Наши рекомендации — это Открытый исходный код, и мы приветствуем PR и проблемы. Однако если ваша отправка изменит эти рекомендации, сначала откройте вопрос для обсуждения. Вы можете использовать наши рекомендации или адаптировать их к вашим потребностям.
Средства и анализаторы
Средства могут помочь вашей команде обеспечить соблюдение соглашений. Вы можете включить анализ кода для применения предпочитаемого правила. Вы также можете создать конфигурацию редактора , чтобы Visual Studio автоматически применяла рекомендации по стилю. В качестве отправной точки можно скопировать dotnet/docs
.editorconfig, чтобы использовать наш стиль.
Эти средства упрощают работу вашей команды по принятию предпочитаемых рекомендаций. Visual Studio применяет правила во всех файлах .editorconfig в области для форматирования кода. Можно использовать несколько конфигураций для применения корпоративных соглашений, соглашений о команде и даже детализированных соглашений о проекте.
Анализ кода создает предупреждения и диагностику при обнаружении нарушений правил. Вы настраиваете правила, примененные к проекту. Затем каждая сборка CI уведомляет разработчиков, когда они нарушают какие-либо правила.
Идентификаторы диагностики
Рекомендации по использованию языка
В следующих разделах описаны рекомендации, которые команда документации .NET следует для подготовки примеров кода и примеров. Как правило, следуйте приведенным ниже рекомендациям.
- По возможности используйте современные языковые функции и версии C#.
- Избегайте устаревших конструкций языка.
- Перехватывайте только те исключения, которые можно правильно обработать; избегайте перехвата общих исключений. Например, пример кода не должен перехватывать тип System.Exception без фильтра исключений.
- Используйте определенные типы исключений для предоставления значимых сообщений об ошибках.
- Используйте запросы и методы LINQ для обработки коллекций, чтобы улучшить удобочитаемость кода.
- Используйте асинхронное программирование с асинхронным и ожиданием операций с привязкой ввода-вывода.
- Будьте осторожны с взаимоблокировками и используйте 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);
Используйте запись в стиле CamelCase для основных параметров конструктора в типах классов и структур.
Используйте свойства
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."); }
Если делитель равен нулю, второе условие в операторе if
вызовет ошибку времени выполнения. При этом оператор && сразу прекращает выполнение, если первое выражение ложно. Это означает, что второе выражение не будет вычисляться. Оператор & всегда вычисляет оба выражения, и если значение 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;
Обработка событий
- Используйте лямбда-выражение для определения обработчика событий, который не требуется удалить позже:
public Form2()
{
this.Click += (s, e) =>
{
MessageBox.Show(
((MouseEventArgs)e).Location.ToString());
};
}
Лямбда-выражение сокращает приведенное ниже традиционное определение.
public Form1()
{
this.Click += new EventHandler(Form1_Click);
}
void Form1_Click(object? sender, EventArgs e)
{
MessageBox.Show(((MouseEventArgs)e).Location.ToString());
}
Статические участники
Для вызова статических членов следует использовать имя класса: 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
более удобочитаемым.Примечание.
Следите за тем, чтобы случайно не изменить тип элемента итерируемой коллекции. Например, легко переключаться с System.Linq.IQueryable на System.Collections.IEnumerable в инструкции
foreach
, которая изменяет выполнение запроса.
Некоторые из наших примеров объясняют естественный тип выражения. Эти примеры должны использовать 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.
Соглашения о макете
Чтобы выделить структуру кода и облегчить чтение кода, в хорошем макете используется форматирование. Примеры и образцы корпорации Майкрософт соответствуют следующим соглашениям.
Использование параметров редактора кода по умолчанию (логичные отступы, отступы по четыре символа, использование пробелов для табуляции). Дополнительные сведения см. в разделе "Параметры", "Текстовый редактор", C#, "Форматирование".
Запись только одного оператора в строке.
Запись только одного объявления в строке.
Если строки продолжения не отступываются автоматически, отступите их на одну остановку табуляции (четыре пробела).
Добавление по крайней мере одной пустой строки между определениями методов и свойств.
Использование скобок для ясности предложений в выражениях, как показано в следующем коде.
if ((startX > endX) && (startX > previousX)) { // Take appropriate action. }
Исключения возникают, когда в примере объясняется приоритет оператора или выражения.
Безопасность
Следуйте указаниям, изложенным в правилах написания безопасного кода.