Рейтинговые книги
Читем онлайн Язык программирования C#9 и платформа .NET5 - Эндрю Троелсен

Шрифт:

-
+

Интервал:

-
+

Закладка:

Сделать
1 ... 26 27 28 29 30 31 32 33 34 ... 407
нет параметров.

static int Main()

{

  // Перед выходом должен возвращать значение!

  return 0;

}

С выходом версии С# 7.1 метод Main() может быть асинхронным. Асинхронное программирование раскрывается в главе 15, но теперь важно помнить о существовании четырех дополнительных сигнатур:

static Task Main()

static Task<int> Main()

static Task Main(string[])

static Task<int> Main(string[])

На заметку! Метод Main() может быть также определен как открытый в противоположность закрытому, что подразумевается, если конкретный модификатор доступа не указан. Среда Visual Studio определяет метод Main() как неявно закрытый. Модификаторы доступа подробно раскрываются в главе 5.

Очевидно, что выбор способа создания метода Main() зависит от ответов на три вопроса. Первый вопрос: нужно ли возвращать значение системе, когда метод Main() заканчивается и работа программы завершается? Если да, тогда необходимо возвращать тип данных int, а не void. Второй вопрос: требуется ли обрабатывать любые предоставляемые пользователем параметры командной строки? Если да, то они будут сохранены в массиве строк. Наконец, третий вопрос: есть ли необходимость вызывать асинхронный код в методе Main()? Ниже мы более подробно обсудим первые два варианта, а исследование третьего отложим до главы 15.

Использование операторов верхнего уровня (нововведение в версии 9.0)

Хотя и верно то, что до выхода версии C# 9.0 все приложения .NET Core на языке C# обязаны были иметь метод Main(), в C# 9.0 появились операторы верхнего уровня, которые устраняют необходимость в большей части формальностей, связанных с точкой входа в приложение С#. Вы можете избавиться как от класса (Program), так и от метода Main(). Чтобы взглянуть на это в действии, приведите содержимое файла Program.cs к следующему виду:

using System;

// Отобразить пользователю простое сообщение.

Console.WriteLine(***** Му First C# Арр *****);

Console.WriteLine("Hello World!");

Console.WriteLine();

// Ожидать нажатия клавиши <Enter>, прежде чем завершить работу.

Console.ReadLine();

Запустив программу, вы увидите, что получается тот же самый результат! Существует несколько правил применения операторов верхнего уровня.

• Операторы верхнего уровня можно использовать только в одном файле внутри приложения.

• В случае применения операторов верхнего уровня программа не может иметь объявленную точку входа.

• Операторы верхнего уровня нельзя помещать в пространство имен.

• Операторы верхнего уровня по-прежнему имеют доступ к строковому массиву аргументов.

• Операторы верхнего уровня возвращают код завершения приложения (как объясняется в следующем разделе) с использованием return.

• Функции, которые объявлялись в классе Program, становятся локальными функциями для операторов верхнего уровня. (Локальные функции раскрываются в главе 4.)

• Дополнительные типы можно объявлять после всех операторов верхнего уровня. Объявление любых типов до окончания операторов верхнего уровня приводит к ошибке на этапе компиляции.

"За кулисами" компилятор заполняет пробелы. Исследуя сгенерированный код IL для обновленного кода, вы заметите такое определение TypeDef для точки входа в приложение:

// TypeDef #1 (02000002)

// -------------------------------------------------------

//   TypDefName: <Program>$  (02000002)

//   Flags     : [NotPublic] [AutoLayout] [Class] [Abstract] [Sealed] [AnsiClass]

     [BeforeFieldInit]  (00100180)

//   Extends   : 0100000D [TypeRef] System.Object

//   Method #1 (06000001) [ENTRYPOINT]

//   -------------------------------------------------------

//          MethodName: <Main>$ (06000001)

Сравните его с определением TypeDef для точки входа в главе 1:

// -------------------------------------------------------

// TypDefName: CalculatorExamples.Program  (02000002)

//   Flags     : [NotPublic] [AutoLayout] [Class] [AnsiClass]

     [BeforeFieldInit]  (00100000)

//   Extends   : 0100000C [TypeRef] System.Object

//   Method #1 (06000001) [ENTRYPOINT]

//   -------------------------------------------------------

//          MethodName: Main (06000001)

В примере из главы 1 обратите внимание, что значение TypDefName представлено как пространство имен (CalculatorExamples) плюс имя класса (Program), а значением MethodName является Main. В обновленном примере, использующем операторы верхнего уровня, компилятор заполняется значение <Program>$ для TypDefName и значение <Main>$ для имени метода.

Указание кода ошибки приложения (обновление в версии 9.0)

Хотя в подавляющем большинстве случаев методы Main() или операторы верхнего уровня будут иметь void в качестве возвращаемого значения, возможность возвращения int (или Task<int>) сохраняет согласованность C# с другими языками, основанными на С. По соглашению возврат значения 0 указывает на то, что программа завершилась успешно, тогда как любое другое значение (вроде -1) представляет условие ошибки (имейте в виду, что значение 0 автоматически возвращается даже в случае, если метод Main() прототипирован как возвращающий void).

При использовании операторов верхнего уровня (следовательно, в отсутствие метода Main()) в случае, если исполняемый код возвращает целое число, то оно и будет кодом возврата. Если же явно ничего не возвращается, тогда все равно обеспечивается возвращение значения 0, как при явном применении метода Main().

В ОС Windows возвращаемое приложением значение сохраняется в переменной среды по имени %ERRORLEVEL%. Если создается приложение, которое программно запускает другой исполняемый файл (тема, рассматриваемая в главе 19), тогда получить значение %ERRORLEVEL% можно с применением свойства ExitCode запущенного процесса.

Поскольку возвращаемое значение передается системе в момент завершения работы приложения, вполне очевидно, что получить и отобразить финальный код ошибки во время выполнения приложения невозможно. Однако мы покажем, как просмотреть код ошибки по завершении программы, изменив операторы верхнего уровня следующим образом:

// Обратите внимание, что теперь возвращается int, а не void.

// Вывести сообщение и ожидать нажатия клавиши <Enter>.

Console.WriteLine("***** My First C# App *****");

Console.WriteLine("Hello World!");

Console.WriteLine();

Console.ReadLine();

// Возвратить произвольный код ошибки.

return -1;

Если программа в качестве точки входа по-прежнему использует метод Main(), то вот как изменить сигнатуру метода, чтобы возвращать int вместо void:

static int Main()

{

   …

}

Теперь давайте захватим возвращаемое значение программы с помощью пакетного файла. Используя проводник Windows, перейдите в папку, где находится файл решения (например, С:SimpleCSharpApp), и создайте в ней новый текстовый файл (по имени SimpleCSharpApp.cmd). Поместите в файл приведенные далее инструкции (если раньше вам не приходилось создавать файлы *.cmd, то можете не беспокоиться о деталях):

@echo off

rem Пакетный файл для приложения SimpleCSharpApp.exe,

rem в котором захватывается возвращаемое им значение.

dotnet run

@if "%ERRORLEVEL%" ==

1 ... 26 27 28 29 30 31 32 33 34 ... 407
На этой странице вы можете бесплатно читать книгу Язык программирования C#9 и платформа .NET5 - Эндрю Троелсен бесплатно.

Оставить комментарий