7  Модульное программирование

Модуль – это  последовательность логически связанных  фрагментов, оформленных как отдельная часть программы.

К модулю предъявляются следующие требования:

1) модуль должен реализовывать единственную функцию, т.е. при построении модуля используется концепция: «один модуль – одна функция». Таким образом, модуль – это элемент программы, выполняющий самостоятельную задачу. На его входе он может получать определенный набор исходных данных, обрабатывать их в соответствии с заданным алгоритмом и возвращать результат обработки, т.е. реализуется стандартный принцип IPO (Input – Process – Output) – вход-процесс-выход;

2) на модуль нужно ссылаться с помощью его имени. Он должен иметь один вход и один выход, что гарантирует замкнутость модуля и упрощает сопровождение программ;

3) модуль должен иметь функциональную завершенность, т.е. выполнять перечень регламентированных операций для реализации каждой отдельной функции в полном составе, достаточных для завершения начатой обработки;

4) модуль должен возвращать управление в точку его  вызова, в свою очередь, он должен иметь  возможность сам вызывать другие модули;

5) модуль не должен сохранять историю своих вызовов и использовать ее при своем функционировании;

6) модуль должен иметь логическую независимость, т.е. результат работы программного модуля зависит только от исходных данных, но не зависит от работы других модулей;

7) модуль должен иметь слабые информационные связи с другими программными модулями – обмен информацией между модулями должен быть по возможности минимизирован;

8) модуль должен быть сравнительно невелик, т.е. быть обозримым по размеру и сложности. Опытные программисты рекомендуют  его размер не более двух страниц распечатки на принтере.

Для  достижения  независимости  модулей  часто используется принцип информационной локализованности, который состоит в том, что вся  информация о  структуре данных, о прототипах функций, констант и т.д. сосредотачивается («упрятывается») в отдельном модуле. Доступ  к этой информации осуществляется только через этот модуль (в алгоритмическом языке С/С++ такие модули имеют расширение *.h).

Программирование с использованием модулей называется модульным программированием. Оно возникло еще в начале 60-х годов XX в. Модульное программирование  основано   на  идее  использования уровней абстракции, когда вся проблема или комплекс задач разбивается на задачи, подзадачи, абстрагируется и представляется в виде иерархического дерева связанных между собой модулей, в совокупности представляющих создаваемое программное обеспечение (ПО).

Достоинствами модульного программирования является следующее:

· большую  программу могут  писать одновременно  несколько программистов, что позволяет раньше закончить задачу;

· можно создавать библиотеки наиболее употребительных модулей;

· упрощается процедура загрузки в оперативную память большой программы, требующей сегментации;

· появляется много естественных контрольных точек для  отладки проекта;

· проще проектировать и в дальнейшем модифицировать программы.

Недостатки модульного программирования заключаеются в следующем:

· возрастает размер требуемой оперативной памяти;

· увеличивается время компиляции и загрузки;

· увеличивается время выполнения программы;

· довольно сложными становятся межмодульные интерфейсы.

Модульное программирование реализуется через модули – функции. Функция – это область памяти, выделяемая для сохранения программного кода, предназначенного для выполнения конкретной задачи. Другими словами, функция – минимальный исполняемый модуль программы на языке С/С++.  По умолчанию функция имеет тип external, и доступ к ней возможен из любого файла программы. Но она может быть ограничена спецификатором класса памяти static.

Функция характеризуется типом, областью действия связанного с функцией имени, видимостью имени функции, типом связывания.

Все функции имеют рекомендуемый стандартами языка единый формат определения. Он имеет заголовок функции, в котором задаются: тип, имя функции и спецификация  формальных параметров:

Тип  имя_функции  (спецификация_параметров) тело_функции

Тип – это тип возвращаемого функцией значения, в том числе void (кроме типов массива или функции). Умолчанием является тип int. Если тип возврата функции не void, то тело функции должно содержать как минимум один оператор return.

Имя_функции – идентификатор, с помощью которого можно обратиться к функции. Он выбирается программистом произвольно и не должен совпадать со служебными словами и с именами других объектов программы. Однако любая программа на языке С/С++ должна иметь хотя бы одну функцию с именем mainглавную функцию, содержащую точку входа в программу.

Спецификация_параметров – список формальных параметров, т.е. переменных, принимающих значения, передаваемые функции при ее вызове. Список формальных параметров перечисляется через запятую. Каждый формальный параметр должен иметь следующий формат:

Тип  имя_формального_параметра

Тип может быть встроенным (int, long, float, double и т.д.), структурой (struct), объединением (union), перечислением (enum), указателями на них или на функции или классы (class). Имя_формального_параметра представляет собой имя используемой в теле функции переменной. Идентификаторы формальных параметров не могут совпадать с именами локальных переменных, объявленных  внутри тела функции.

Объявление формального параметра может содержать инициализатор, то есть выражение, которое должно обеспечить параметру присвоение начального значения. Инициализатор параметра не является константным выражением. Начальная инициализация параметров происходит не на стадии компиляции (как, например, выделение памяти под массивы), а непосредственно в ходе выполнения программы.

В языке С/C++ допустимы функции, количество параметров у которых при компиляции функции не фиксировано, следовательно, остаются неизвестными и их типы. Количество и типы параметров таких функций становятся известными только при их вызове, когда явно задан список фактических параметров. При определении и описании таких функций со списками параметров неопределенной длины спецификацию формальных параметров следует закончить запятой и многоточием.

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

Спецификация_параметров может отсутствовать, то есть скобки могут быть пустыми, но в этом случае рекомендуется указывать тип void.

Тело_функции – часть определения функции, ограниченная фигурными скобками и непосредственно размещенная вслед за заголовком функции. Тело_функции может быть либо составным оператором, либо блоком. Например:

double f (int n, float x)

{

/*Тело функции*/

}

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

Тип  имя_функции  (спецификация_параметров);

В отличие от заголовка функции в прототипе функции имена формальных параметров могут не указываться. Например, следующие прототипы одной и той же функции эквивалентны:

double func (int n, float x);

double func (int, float);

Для обращения к функции используется выражение с операцией «круглые скобки»:

Имя_функции  (список_фактических_параметров);

Операндами операции «круглые скобки» служат имя_функции и список_фактичес­ких_параметров, т.е. список выражений, количество которых равно числу формальных аргументов функции (за исключением функций с переменным количеством параметров). Между формальными и фактическими параметрами должно быть соответствие по типам, т.е. тип формального параметра должен соответствовать типу фактического параметра.

Фактические параметры могут быть любыми значениями того же типа, что и формальные параметры, массивом или указателем на любой тип или функцию. Все фактические параметры передаются по значению. Их значения копируются в соответствующие формальные параметры. Функция использует только копии объектов, не изменяя самих объектов. Компилятор выполняет преобразования по умолчанию для каждого типа формального параметра и каждого фактического аргумента.

Функция может не иметь фактических параметров и не возвращать никакого значения, например:

#include  <stdio.h>

void Real_Time  (void)

{

printf  ("n  Текущее  время :  %s", __TIME__

"  (час:  мин:  с) " );

}

При обращении к функции

Real_Time  (  );

в результате выполнения функции будет выведено на экран сообщение:

Текущее  время:   14:16:25    (час:  мин:  с)

Для оказания помощи программистам язык С предлагает готовые функции, организованные в виде библиотеки и распределенные на несколько групп. Каждая группа имеет свой заголовочный файл, подключаемый с помощью директивы include. В приложении 2 приведены некоторые часто употребляемые функции, объединенные в такие группы.