Язык программирования C#9 и платформа .NET5
Очевидно, что выбор способа создания метода
зависит от ответов на три вопроса. Первый вопрос: нужно ли возвращать значение системе, когда методMain()заканчивается и работа программы завершается? Если да, тогда необходимо возвращать тип данныхMain(), а неint. Второй вопрос: требуется ли обрабатывать любые предоставляемые пользователем параметры командной строки? Если да, то они будут сохранены в массиве строк. Наконец, третий вопрос: есть ли необходимость вызывать асинхронный код в методеvoid? Ниже мы более подробно обсудим первые два варианта, а исследование третьего отложим до главы 15.Main()Использование операторов верхнего уровня (нововведение в версии 9.0)
Хотя и верно то, что до выхода версии C# 9.0 все приложения .NET Core на языке C# обязаны были иметь метод
, в C# 9.0 появились операторы верхнего уровня, которые устраняют необходимость в большей части формальностей, связанных с точкой входа в приложение С#. Вы можете избавиться как от класса (Main()), так и от методаProgram. Чтобы взглянуть на это в действии, приведите содержимое файлаMain()к следующему виду:Program.csusing System;// Отобразить пользователю простое сообщение.Console.WriteLine(***** Му First C# Арр *****);Console.WriteLine("Hello World!");Console.WriteLine();// Ожидать нажатия клавиши <Enter>, прежде чем завершить работу.Console.ReadLine();Запустив программу, вы увидите, что получается тот же самый результат! Существует несколько правил применения операторов верхнего уровня.
• Операторы верхнего уровня можно использовать только в одном файле внутри приложения.
• В случае применения операторов верхнего уровня программа не может иметь объявленную точку входа.
• Операторы верхнего уровня нельзя помещать в пространство имен.
• Операторы верхнего уровня по-прежнему имеют доступ к строковому массиву аргументов.
• Операторы верхнего уровня возвращают код завершения приложения (как объясняется в следующем разделе) с использованием
.return• Функции, которые объявлялись в классе
, становятся локальными функциями для операторов верхнего уровня. (Локальные функции раскрываются в главе 4.)Program• Дополнительные типы можно объявлять после всех операторов верхнего уровня. Объявление любых типов до окончания операторов верхнего уровня приводит к ошибке на этапе компиляции.
"За кулисами" компилятор заполняет пробелы. Исследуя сгенерированный код IL для обновленного кода, вы заметите такое определение
для точки входа в приложение:TypeDef// TypeDef #1 (02000002)// -------------------------------------------------------// TypDefName: <b><Program>$</b> (02000002)// Flags : [NotPublic] [AutoLayout] [Class] [Abstract] [Sealed] [AnsiClass][BeforeFieldInit] (00100180)// Extends : 0100000D [TypeRef] System.Object// Method #1 (06000001) [ENTRYPOINT]// -------------------------------------------------------// <b>MethodName: <Main>$ (06000001)</b>Сравните его с определением
для точки входа в главе 1:TypeDef// -------------------------------------------------------// TypDefName: <b>CalculatorExamples.Program</b> (02000002)// Flags : [NotPublic] [AutoLayout] [Class] [AnsiClass][BeforeFieldInit] (00100000)// Extends : 0100000C [TypeRef] System.Object// Method #1 (06000001) [ENTRYPOINT]// -------------------------------------------------------// <b>MethodName: Main (06000001)</b>В примере из главы 1 обратите внимание, что значение
представлено как пространство имен (TypDefName) плюс имя класса (CalculatorExamples), а значениемProgramявляетсяMethodName. В обновленном примере, использующем операторы верхнего уровня, компилятор заполняется значениеMainдля<Program>$и значениеTypDefNameдля имени метода.<Main>$Указание кода ошибки приложения (обновление в версии 9.0)
Хотя в подавляющем большинстве случаев методы
или операторы верхнего уровня будут иметьMain()в качестве возвращаемого значения, возможность возвращенияvoid(илиint) сохраняет согласованность C# с другими языками, основанными на С. По соглашению возврат значенияTask<int>указывает на то, что программа завершилась успешно, тогда как любое другое значение (вроде0) представляет условие ошибки (имейте в виду, что значение-1автоматически возвращается даже в случае, если метод0прототипирован как возвращающийMain()).voidПри использовании операторов верхнего уровня (следовательно, в отсутствие метода
) в случае, если исполняемый код возвращает целое число, то оно и будет кодом возврата. Если же явно ничего не возвращается, тогда все равно обеспечивается возвращение значенияMain(), как при явном применении метода0.Main()В ОС Windows возвращаемое приложением значение сохраняется в переменной среды по имени
. Если создается приложение, которое программно запускает другой исполняемый файл (тема, рассматриваемая в главе 19), тогда получить значение%ERRORLEVEL%можно с применением свойства%ERRORLEVEL%запущенного процесса.ExitCode