Функции


Тело

[return [РВ];]

}

 

 

где имя – идентификатор (название) функции;

тип – описатель типа функции (результата);

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

описатели bi – операторы описания типов формальных параметров, выполненные вне заголовка;

( ) – ограничители списка формальных параметров;

тело функции – основная часть (совокупность операторов), реализующая вынесенные в отдельный алгоритм вычисления (действия);

return РВ; – оператор возврата (return) результата вычислений;

РВ – результат вычислений (выражение);

[ ] – признак необязательности содержимого;

{ } – ограничители тела функции.

Первая строка структуры является заголовком функции.

Структура вызова функции:

имя ([a1, . . . , ai, . . . , an])

где имя – идентификатор функции;

a1, ai, an – список (перечисление через запятую) фактических параметров (аргументов), численные значения которых требуется передать в дополнительную функцию (подпрограмму);

( ) – ограничители аргументов;

[ ] – признак необязательности содержимого.

Правила записи и использования

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

2. Деление функций на главную и дополнительные осуществляется по имени. Главной является функция с именем main (головная). В этой функции в большинстве случаев описатель типа, список формальных параметров и оператор return не указываются.

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

4. Функция предписывает выполнение указанных в теле вычислений (действий) с операндами – формальными параметрами, идентификаторы которых указаны в списке заголовка.

5. В качестве формальных параметров используются переменные и указатели.

6. Физически формальные параметры (переменные) – дополнительные ячейки памяти, предназначенные для временного хранения, передаваемых им значений фактических параметров.

7. Количество, типы и последовательность расположения формальных параметров в списке задается создателем функции.

8. В функции наряду с формальными параметрами могут использоваться вспомогательные (локальные) переменные.

9. Для возврата результата вычисления тела функции дополняются выражением оператора return. В качестве выражения могут использоваться константы, переменные и их совокупности. Результаты вычислений тела функции – константы.

10. В основную (вызывающую) функцию передается значение только одной константы – результата вычисления выражения return. Поэтому, если все планируемые в алгоритме вычисления могут быть сведены к одной формуле, их можно оформить в качестве выражения return.

11. Тип возвращаемого значения определяется типом функции. Если он не указан, подразумевается целочисленный результат (int). Результат возвращается в точку вызова.

12. В функции при необходимости могут быть несколько операторов возврата (return), располагаемых в теле функции в соответствии с требованиями алгоритма.

13. В головной функции оператор return можно не указывать (результат не требуется возвращать куда-либо). В этом случае на этапе компиляции может выдаваться соответствующее предупреждение.

14. Головная и дополнительные функции – есть отдельные программные модули. Поэтому в них могут использоваться одноимённые переменные, указатели и массивы.

15. Тексты (определения) дополнительных пользовательских функций могут располагаться в любом месте программы (вне тела любой из функций), стандартных – в соответствующих библиотеках языка Си в виде объектных модулей.

16. Независимо от взаиморасположения функций выполнение программного модуля начинается с головной функции.

17. Из тела каждой функции можно обращаться к любой другой, а также и к ней самой.

18. Фактические и формальные параметры в списках должны совпадать по количеству, месту расположения и типу.

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

20. Вызов дополнительной функции требует ее предварительного определения (расположение над вызывающей). В противном случае её необходимо описать (задать прототипом). Описание размещается над вызывающей функцией. Описания стандартных функций (их прототипы) находятся в стандартных заголовочных файлах.

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

Прототип функции аналогичен ее заголовку и имеет структуру

[тип] имя([тип [b1], . . . ,тип [bi], . . . ,тип [bn]]);

где имя – идентификатор (название) функции;

тип – описатель типа функции (результата);

b1, bi, bn – список (перечисление через запятую) формальных параметров с указанием типа;

( ) – ограничители списка формальных параметров;

[ ] – признак необязательности содержимого;

; – признак оператора.

В списке допускается указывать только типы формальных параметров.

Рассмотрим конкретные примеры определения, вызова, и описания функций.

Пример взаимодействия головной функции с одной пользовательской и стандартными

Вариант 1.

#include<stdio.h> /* stdio.h - файл с прототипами функций ввода-вывода */

#include<math.h> /* math.h - файл с прототипами математических функций */

 

float sum(float a, float b); /* прототип пользовательской функции */

 

main( ) /* заголовок головной функции */

{

float a, b, c, d, d1; /* описатели локальных переменных */

. . .

c = pow( a, 2 ) + b; /* вычисление с вызовом

стандартной функции pow */

d = sum( a, b ) - 0.5*c; /* вычисления с вызовами поль- */

d1 = d + sum( 3.6, sqrt(d) ); /* зовательской функции sum */

. . .

printf(“%f %f %f ”, c, d, d1); /* вызов стандартной функции printf */

}

 

/* определение пользовательской функции sum */

float sum(float s1, float s2 ) /* заголовок дополнительной функции */

{

float s; /* описание локальной переменной s */

s = s1 + s2; /* вычисление s по формальным параметрам s1 и s2 */

return s; /* возвращение значения s в вызывающую функцию */

}

В представленном фрагменте до определения главной функции, путем подключения заголовочных файлов, выполнены описания стандартных функций (математических и ввода-вывода) и пользовательской функции sum – указанием её прототипа. Главная функция расположена над дополнительной и не содержит в заголовке описатель типа и список формальных параметров. Дополнительная функция предназначена для вычисления суммы двух операндов (формальных параметров) и выполнена по первому варианту структуры. Заголовок определяет тип возвращаемого значения (float), название функции (sum) и список формальных параметров – вещественных переменных s1 и s2. Порядок их расположения в списке задан произвольно. Искомая переменная s (локальная) рассчитывается в теле функции и указана в операторе return как возвращаемое значение. Из тела главной функции выполняются обращения к стандартным функциям: sqrt – с одним фактическим параметром (d), pow – с двумя (a, 2) и printf – с четырьмя (“%f %f %f ”, c, d, d1). В арифметическом выражении вычисления переменной d в качестве первого операнда используется вызов пользовательской функции sum с указанием фактических параметров – a и b, описанных в головной функции как локальные переменные. Типы (вещественные) и последовательность расположения в списке определяются аналогичными формальным параметрам (s1 и s2) заголовка вызываемой функции sum.

В арифметическом выражении вычисления переменной d1 в качестве второго операнда используется вызов пользовательской функции sum с указанием фактических параметров – 3.6 и sqrt( d ). Типы операндов (вещественные) и последовательность расположения в списке определяются аналогичными формальным параметрам (s1 и s2) заголовка вызываемой функции sum.

Выполнение программного модуля (аналогично работе со стандартными функциями) приведет к двукратной автоматической передаче численных значений фактических параметров (a, b), а затем (3.6, sqrt(d)) из головной функции в вызываемую пользовательскую (sum) формальным параметрам (s1, s2). После выполнения вычислений тела функции результат (s) автоматически возвратится в точки вызова как операнды вычисления переменных d и d1. Время существования переменных s1, s2 и s соответствует длительности работы функции sum.

Вариант 2.

#include<stdio.h> /* stdio.h - файл с прототипами функций ввода-вывода */

#include<math.h> /* math.h - файл с прототипами математических функций */

 

/* определение пользовательской функции sum */

float sum(s1, s2 ) /* заголовок дополнительной функции */

float s1, s2;

{return s2+s1; /*возвращение значения s в вызывающую функцию */

}

main( ) /* заголовок головной функции */

{ float a, b, c, d, d1; /* описатели локальных переменных */

. . .

c = pow( a, 2 ) + b; /* вычисление с вызовом

стандартной функции pow */

d = sum( a, b ) - 0.5*c; /* вычисления с вызовами поль- */

d1 = d + sum( 3.6, sqrt(d) ); /* зовательской функции sum */

. . .

printf(“%f %f %f ”, c, d, d1); /* вызов стандартной функции printf */

}

Дополнительная функция составлена с использованием второго варианта структуры. Тело дополнительной функции составляет один оператор return.

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

Рассмотренные структуры и правила их использования позволяют осуществить программирование задачи с факториалами, предварительно выполнив идентификацию переменных (табл. 6.1).

Таблица 6.1

Обозначение в алгоритме a z n m k f fi i
Обозначение в программе a z n m k f fi i

Программа решения примера 6.1

#include <stdio.h> /* stdio.h - файл с прототипами функций ввода-вывода */

#include <conio.h> /* conio.h - файл с прототипом функций getch( ), clrscr( )*/

#include <math.h> /* math.h - файл с прототипами математических функций*/

long fakt(int fi); /* прототип функции расчета факториала */

/* определение головной функции */

main( )

{ float a, z1, z2, z;

int n, m, k;

clrscr( );

printf("\n Введите значения a, n, m, k: ");

scanf("%f%d%d%d", &a, &n, &m, &k);

z1 = log(a) + fakt(n);

z2 = fakt(m) + fakt( k-m );

z = z1 + z2 - 0.2*a;

fprintf(stdprn, "\n a=%.2f n=%d m=%d k=%d"

"\n z1=%.2f z2=%.2f"

"\n z=%.2f\n", a, n, m, k, z1, z2, z);

getch( );

}

/* определение функции расчета факториала */

long fakt(int f)

{int i;

long fi; /* описание переменной текущего значения факториала */

fprintf(stdprn, "\n\n f=%d", f);

for( fi=1, i=1 ; i<=f ; i++ ) /* заголовок цикла расчета факториала */

{fi = fi * i; /* расчет текущего значения факториала */

fprintf(stdprn, "\n %d %ld",i, fi); /* вывод текущего значения */

}

return fi; /* возврат в головную функцию значения переменной fi */

}

16.9 5 4 10

В подпрограмме наряду с формальным параметром f использованы локальные переменные fi и i. Первая из них определяет значение вычисляемого факториала, изменяясь от единицы до конечного значения. Вторая используется как параметр цикла для организации последовательного умножения в заданном диапазоне. Функция fprintf позволяет проверить правильность работы подпрограммы.