Программные объекты
В C++ любая программа состоит из точки входа (в ANSI C++ это функция main()
, хотя для приложений Windows она обычно называется WinMain()
), а также различных классов. структур и глобальных переменных или функций, которые определены вне любого класса. Хотя многие разработчики будут считать что хороший объектно-ориентированный проект определяется тем, насколько возможно, чтобы элементы самого верхнего уровня в коде являлись объектами C++ не требует этого. Как только что было показано, C# реализует эту идею. Он утверждает существенно более объектно- ориентированную парадигму, требуя, чтобы все элементы являлись членами класса. Другими словами, единственными объектами верхнего уровня в программе являются классы (или другие элементы, которые могут рассматриваться как специальные типы классов: перечисления, делегаты и интерфейсы). В этом случае код C# оказывается более объектно-ориентированным, чем это требует C++.
Файловая структура
В C++ синтаксис, на котором строится программа, основывается на понятии файла как единице исходного кода. Имеются, например, файлы исходного кода (файлы .срр
), каждый из которых будет содержать директивы препроцессора #include
для включения подходящих заголовочных файлов. Процесс компиляции содержит отдельную компиляцию каждого исходного файла, после чего эти файлы объектов компонуются для создания конечного исполнимого файла. Хотя конечный исполнимый файл не содержит никакой информации о первоначальных исходных или объектных файлах, C ++ был создан таким образом, что разработчик явно кодирует в рамках выбранной структуры файлов исходного кода.
При использовании C# в ведении компилятора находятся детали соответствия отдельных файлов исходного кода. Можно поместить исходный код в один файл или в несколько файлов, но это не играет никакой роли для компилятора и нет необходимости для любого файла явно ссылаться на другие файлы. В частности, не существует требования определять элементы до ссылки на них в любом отдельном файле, как это требуется в C++. Компилятор успешно находит определение каждого элемента, где бы оно не находилось. Как побочный эффект этого, не существует в действительности никакой концепции компоновки кода в C#. Компилятор просто компилирует все исходные файлы в сборку (хотя можно определить другие варианты, к примеру, модуль — единица, которая будет формировать часть сборки). Компоновка не имеет места в C#, но она в действительности ограничивается компоновкой кода с любым существующим библиотечным кодом в сборках. В C# не существует такого понятия, как заголовочный файл.
Точка входа программы
В стандартном ANSI C++ точка входа программы является по умолчанию функцией с именем main()
, которая имеет сигнатуру:
int main(int argc, char *argv)
Здесь argc указывает число аргументов, передаваемых в программу, a argv является массивом строк, задающих эти аргументы. Первый аргумент всегда является командой, используемой для выполнения самой программы. Windows несколько изменяет это. Приложения Windows традиционно начинаются с точки входа, называемой WinMain()
, a DLL с DllMain()
. Эти методы также получают другие множества параметров.
В C# точка входа следует аналогичным принципам. Однако в связи с требованием, что все элементы C# являются частью класса, точка входа не может быть глобальной функцией. Вместо этого, как было сказано ранее, требуется, чтобы один из классов имел статический метод-член с именем Main()
.
Синтаксис языка
C# и C++ используют практически идентичный синтаксис. Оба языка, например, игнорируют пробелы между инструкциями и используют точку с запятой для разделения инструкций и фигурные скобки для объединения инструкций в блоки. Все это означает, что на первый взгляд программы, написанные любым языком, выглядят очень похожими. Отметим, однако, следующие различия:
□ C++ требует точку с запятой после определения класса. C# не требует.
□ C++ позволяет использовать выражения как инструкции, даже если они не имеют результата, например:
i + 1;
В C# это будет ошибкой.
Необходимо также отметить, что подобно C++, C# различает строчные и заглавные символы. Однако, так как C# создан для взаимодействия с VB.NET (который не отличает такие символы), строго рекомендуется не использовать имена, которые разнятся только регистром символов для каких-либо объектов, видных коду вне данного проекта (другими словами, имена открытых членов классов в коде библиотеки). Если используются открытые имена, которые отличаются только регистром символов, то это не позволит коду VB.NET получить доступ к этим классам. (В случае написания какого-либо управляемого кода C++ для среды .NET применимы те же рекомендации.)
Опережающие объявления
Опережающие объявления не поддерживаются и не требуются в C#, так как порядок, в котором элементы определены в файлах исходного кода, не имеет значения. Вполне допустимо одному элементу ссылаться на другой элемент, который позже определяется в этом файле или в другом файле, он должен только где-то быть определен. Это противоположно C++, в котором на символы и т.д. можно ссылаться в любом из файлов исходного кода, если они уже были объявлены в том же файле или во включаемом файле.
Отсутствие разделения определения и объявления
С отсутствием опережающих объявлений в C# связано то, что в C# не существует никакого разделения объявления и определения для любых элементов. Например, в C++ принято описывать класс в заголовочном файле сигнатурой функций членов, а полные определения даны в другом месте:
class CMyClass {
public:
void MyMethod(); // определение этой функции находится в файле C++,
// если только MyMethod() не является встраиваемой функцией