Добавил:
СПбГУТ * ИКСС * Программная инженерия Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Язык Си. Лабораторные работы / Справочник. Часть 2 (СПбГУТ).doc
Скачиваний:
49
Добавлен:
10.09.2019
Размер:
801.79 Кб
Скачать

2.6.2. Функция как обобщенное решение ряда частных задач

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

Пример. Формульный счет.

Постановка задачи. Вычислить значение величины y:

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

В обобщенной формуле имеются два параметра: x1 и x2. Подставим в эту формулу значения параметров x1 == a – 1 и x2 == x. Это позволит вычислить значение первого из корней .

Подстановка: x1 == x и x2 == 1 в обобщенную формулу позволит вычислить значение второго корня .

Наконец, подставляя значения x1 == a и x2 == 2, получим последний из корней.

Программный код решения задачи.

#include<stdio.h> #include<math.h> double sqrt_sum_sqr(double x1, double x2); int main(void) { double a; printf(“%s”, “a=”); scanf(“%lf”, &a); double x; printf(“%s”, “x=”); scanf(“%lf”, &x); double y = (sqrt_sum_sqr(a – 1, x) + sqrt_sum_sqr(x, 1) + sqrt_sum_sqr(a, 2)) / 3; printf(“y = %0.4g\n”, y); getchar(); return 0; } /* Функция для вычисления корня из суммы квадратов*/ double sqrt_sum_sqr(double x1, double x2) { return sqrt(x1 * x1 + x2 * x2); }

2.6.3. Параметры функции или глобальные переменные

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

Постановка задачи. Вычислить значение величины y, заданной следующим образом.

Решение. Вариант 1. Рекомендуемый вариант. Внешние переменные не используются.

Анализ расчетной формулы показывает, что она содержит три подвыражения, имеющие одинаковую структуру.

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

Нетрудно видеть, что функция, выполняющая вычисления по приведенной выше обобщенной формуле, должна иметь четыре параметра. Три из них (i_min, i_max и x) введены в процессе получения обобщенной формулы. Они учитывают индивидуальные особенности подвыражений. Четвертый параметр (n) одинаков для всех трех индивидуальных подвыражений. Это обстоятельство будет учтено при рассмотрении второго варианта решения этой задачи. Используя обобщенное выражение, можно написать следующий прототип разрабатываемой функции.

Double root_summa(int i_min, int i_max, double x, int n);

Наличие функции с таким прототипом позволяет написать для вычисления искомой величины y следующую инструкцию языка Си.

Y = (root(3, m1, -a, n) + root(2, m2, 0, n)) / (4 + root(2, m1, 2, n));

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

#include<stdio.h> #include<math.h> double root_summa(int i_min, int i_max, double x, int n); int main(void) { int n; printf(“%s”, “n=”); scanf(“%d”, &n); double a; printf(“%s”, “a=”); scanf(“%lf”, &a); int m1; printf(“m1=”); scanf(“%d”, &m1); int m2; printf(“m2=”); scanf(“%d”, &m2); double y = (root_summa(3, m1, -a, n) + root_summa(2, m2, 0, n)) /

(4 + root_summa(2, m1, 2, n)); printf(“y = %0.4g\n”, y); getchar(); return 0; } /* Функция для вычисления корня n степени из суммы */ double root_summma(int i_min, int i_max, double x, int n) { double s = 0; for(int i = i_min; i <= i_max; i++) s += (i + x) * (i + x); return pow(s, 1.0 / n); }

Вариант 2. Замена параметров функции внешними переменными.

Анализ решения рассматриваемой задачи, приведенный в варианте 1, показывает, что параметр (n) во всех трех вызовах функции root_summa() имеет одно и то же значение. Есть основания заменить такой параметр внешней переменной. После выполнения замены программа принимает следующий вид.

#include<stdio.h> #include<math.h> double root_summa_2(int i_min, int i_max, double x); int n; //Внешняя переменная int main(void) { <ввод n> <объявление и ввод a, m1 и m2> double y = (root_summa_2(3, m1, -a)+ root_summa_2(2, m2, 0))/ (4 + root_summa_2(2, m1, 2)); printf(“y = %0.4g\n”, y); getchar(); return 0; } /* Функция для вычисления корня n степени из суммы n – внешняя переменная целого типа, равная степени корня */ double root_summma_2(int i_min, int i_max, double x) { <тело идентично телу функции root_summa> }

Сравнивая два полученных решения рассматриваемой задачи, начинающий программист может отдать предпочтение второму варианту. Действительно, функция root_summa_2() имеет меньшее количество параметров по сравнению с функцией root_summa(). Это уменьшает накладные расходы, которые имеют место при вызове функции. Наличие внешней единственной переменной не сильно будет ухудшать ее читабельность. Дело в том, что сама программа невелика. В программе имеется комментарий, который предупреждает, что функция root_summa_2() использует внешнюю переменную. В программе имеется только одна функция. Поэтому невелика опасность случайного (при наличии в них ошибок) изменения внешней переменной. Неустранимым недостатком использования внешних переменных является ухудшение повторного использования кода. Предположим, что разработанную при решении рассматриваемой задачи функцию, необходимо перенести в другую программу. При переносе функции root_summa() программисту достаточно скопировать текст этой функции и перенести его в новую программу; при переносе функции root_summa_2()ему придется считаться с наличием внешней переменной n. Её придётся переносить отдельно. Могут возникнуть дополнительные трудности, когда новая программа уже содержит внешнюю переменную с именем n. Трудности будут нарастать, когда программа будет содержать несколько функций, в которых используются внешние переменные.

Вариант 3. Неудачный вариант. Полная замена параметров функции внешними переменными.

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

#include<stdio.h> #include<math.h> double root_summa_3(void); int n, i_min, i_max;//Внешние переменные double x; //Внешняя переменная int main(void) { <ввод исходных данных n, a, m1 и m2> // Подготовка к первому вызову функции root_summa_3 i_min = 3; i_max = m1; x = -a; double s1 = root_summa3(); // вычисление 1–ой суммы //Подготовка ко второму вызову функции root_summa_3 i_min = 2; i_max = m1; x = 0; double s2 = root_summa_3();// вычисление 2–ой суммы // Подготовка к третьему вызову функции root_summa_3 i_min = 2; i_max = m1; x = 2; double y = (s1 + s2) / (4 + root_summa_3()); printf(“y = %0.4g\n”, y); getchar(); return 0; } //Функция для вычисления корня n степени из суммы //Функция использует четыре внешних переменных: //int n – степень корня; int i_min и i_max – минимальное и // максимальное значение переменной суммирования; // double x – параметр, входящий в слагаемые суммы double root_summa_3(void) { <тело данной функции идентично телу функции root_summa> }

Приведенное решение свидетельствует о существенном усложнении клиентского кода функции при замене параметров внешними переменными.