bmstu_iu6_Cpp1 (823967), страница 15
Текст из файла (страница 15)
Использование указателя на структуру в качестве параметра функции.При использовании ссылки на структуру изменится текст функции и ее вызов.Текст функции:int summa(struct mas &x) // передается ссылка на структуру{int i,s=0;for (i=0;i<x.n;i++)x.sum=s;s+=x.a[i]; // суммирование элементов// запись суммы элементов в поле структурыreturn s;}Вызов:printf("Сумма элементов=%4d.\n",summa(massive));Однако следует понимать, что при использовании массива структур имя массива является указателем, который хранит адрес структуры.Пример 4.7. Программа определяет сумму элементов каждой из трех структур, записывает эту сумму в соответствующее поле структуры и определяет общую сумму.#include <locale.h>#include <stdio.h>#include <conio.h>struct mas{int n, a[10], sum;};int summa(struct mas x[]);int main(int argc, char* argv[]){setlocale(0,"russian");int i,k;struct mas massive[3];Оглавление96for (k=0;k<3;k++){puts("Введите количество элементов:");scanf("%d",&massive[k].n);puts("Введите элементы:");for (i=0;i<massive[k].n;i++)scanf("%d",&massive[k].a[i]);}printf("Сумма элементов=%4d.\n",summa(massive));puts("Нажмите любую клавишу для завершения...");_getch();return 0;}int summa(struct mas x[])// в функцию передается указатель на массив{int i,k,s,s1=0;for (k=0;k<3;k++,x++)/* x++ - осуществляет переадресацию наследующий элемент структуры */{for (s=0,i=0;i<x->n;i++)s+=x->a[i];x->sum=s;s1+=s;}return s1;}4.1.3 * Рекурсивные функцииРекурсивная подпрограмма подразумевает организацию вычислений, при которойпроцедура или функция обращается к самой себе.Рассмотрим несколько примеров.Пример 4.8.
Вычисление наибольшего общего делителя. Линейная рекурсия#include <locale.h>#include <stdio.h>Оглавление97#include <conio.h>int nod(int a,int b){if(a==b) return a;// базисное утвержлениеelse{if (a>b) return nod(a-b,b);// рекурсивное утверждениеelse// рекурсивное утверждениеreturn nod(a,b-a);}}int main(int argc, char* argv[]){setlocale(0,"russian");int a,b;puts("Введите два целых числа:");scanf("%d %d",&a,&b);printf("\nНаибольший общий делитель %5d%5d = %5d\n",a,b,nod(a,b)); // вызов функцииputs("Нажмите любую клавишу для завершения...");_getch();return 0;}Пример 4.9.
Вычисление n-го числа Фибоначчи. Древовидная рекурсия#include <locale.h>#include <stdio.h>#include <conio.h>int fib(int n){if((n==1)||(n==2)) return 1; // базисное утверждениеelse{return fib(n-1) + fib(n-2); // рекурсивное утверждение}}Оглавление98int main(int argc, char* argv[]){setlocale(0,"russian");int n;puts("Введите номер элемента Фибоначчи:");scanf("%d",&n);printf("\n %7d Номер =",n);printf("%10d\n",fib(n));// Вызов рекурсивной функцииputs("Нажмите любую клавишу для завершения...");_getch();return 0;}4.1.4 * Дополнительные возможности функций С++При создании С++ разработчики несколько расширили возможности по организациифункций по сравнению с теми, которые использовались в Си. Так появились подставляемые функции, параметрическая перегрузка функций и функции с параметрами, задаваемыми по умолчанию.1.Подставляемые функции.
Вставляемыми или подставляемыми функциямив С++ называются функции, код которых вставляется в то место программы, где они вызываются. Такие функции используют для сокращения времени вызова подпрограммы, таккак в этом случае не выполняется процедура вызова функции и обратной передачи управления. Естественно подставляемые функции не должны быть большими, а то используемый прием потеряет смысл.Для обозначения подставляемых функций используется служебное слово inline,например функция, возвращающая абсолютное значение параметра:inline int abs(int a) {return a>0?a:-a;)}Традиционно на использование подставляемых функций накладываются ограничения. Такая функция:• не должна содержать циклов;• не может быть рекурсивной или виртуальной;• не должна вызываться более одного раза в выражении;• не должна вызываться до своего определения (а не до объявления, как обычнаяфункция).Оглавление99Если вставка не возможна, то служебное слово inline игнорируется, функция компилируется независимо, а при вызове используется стандартный механизм подключения.В процессе оптимизации компилятор Microsoft Visual C++ самостоятельно принимает решение о вставке или вызове объявленных inline функций, поэтому он никаких предупреждающих сообщений не выдает.2.
Переопределяемые функции. В С++ функции различаются по списку, количествуи типам параметров. Поэтому было разрешено определять несколько вариантов одной итой же функции с одинаковыми именами, но с разными списками параметров. При вызове,компилятор по списку аргументов определяет нужный аспект функции и вызывает требуемую ее реализацию.int lenght(int x,int y){return sqrt(x*x+y*y);}int lenght(int x,int y,int z){return sqrt(x*x+y*y+z*z);}3.
Параметры функции, принимаемые по умолчанию. С++ предоставляет возможность при определении функции присваивать значения некоторым параметрам. Этизначения параметры будут принимать по умолчанию, если при вызове функции соответствующие параметры не будут указаны. Однако такие параметры должны располагаться вконце списка параметров, поскольку пропускать параметры при вызове функции не разрешается.Значение по умолчанию задается с помощью синтаксической конструкции, очень похожей на инициализацию переменной, например:void InitWindow(int xSize=80,int ySize=25,int barColor=BLUE,int FrameColor=CYAN){...}Теперь функцию можно вызвать, пропуская часть параметров, например:InitWindow(); // все параметры берутся по умолчаниюInitWindow(20,10); // меняются размеры окна, остальные – по умолчаниюНо, если нужно изменить, например, цвет, оставив остальные параметры неизменными, то все предыдущие параметры следует повторить.InitWindow(80,25,GREEN); // меняем цвет рамки окна, остальные –// по умолчаниюInitWindow(80,25,BLUE,GREEN); // меняем цвет фона окна, остальные –// по умолчаниюОглавление1004.2 Модули С++Среда Visual C++ позволяет создавать и отлаживать программы, использующие нетолько стандартные, но и пользовательские библиотеки подпрограмм – модули.
Модуль C++ обычно включает два файла: заголовочный файл с расширением «.h» и файл реализациис расширением «.cpp».Заголовочный файл играет роль интерфейсной секции модуля. В него помещают объявление экспортируемых ресурсов модуля:• прототипы (заголовки) процедур и функций,• объявление переменных, типов и констант.Заголовочный файл подключают командой #include ″<Имя модуля>.h″, записываемой в файле реализации программы или другого модуля, если они используют ресурсыописываемого модуля.Файл реализации представляет собой секцию реализации модуля. Он должен содержать команды подключения используемых модулей, описания экспортируемых процедур ифункций, а также объявление внутренних ресурсов модуля. Файл реализации подключается командой #include ″<Имя модуля>.cpp″.При создании первый (главный) файл проекта уже содержит заготовку основнойфункции программы – функции main().
Для создания файлов модуля и добавления их кпроекту необходимо вновь вызвать многошаговый Мастер заготовок. Это делается с использованием команды меню File/New. Выполнение этой команды при открытом проектевызовет открытие окна Мастера заготовок на вкладке Files, на которой необходимо выбрать тип файла, добавляемого к проекту.Пример 4.10. Разработать модуль для нахождения наибольшего общего делителядля двух целых чисел.Файл Mod.h:#ifndef modh_20100810#define modh_20100810int nod(int a,int b);#endifДополнительные команды препроцессора в файле mod.h позволяют исключить повторную компиляцию текста программы при многократном подключении заголовочногофайла (см.
раздел 6.3).Оглавление101Имя переменной препроцессора modh_20100810 – уникальный идентификатор,который точно не встречается в других библиотеках. Для получения этого имени в примере для mod.h использована следующая схема:<ИМЯФАЙЛА><РАСШИРЕНИЕ>_<ГОД><МЕСЯЦ><ДЕНЬ (создания)>.Файл Mod.cpp:#include "Mod.h"int nod(int a,int b){while (a!=b)if (a>b) a=a-b; else b=b-a;return a;}Файл main.cpp:#include <locale.h>#include <stdio.h>#include <conio.h>#include "mod.h"int main(int argc, char* argv[]){setlocale(0,"russian");int a=18,b=24,c;c=nod(a,b);printf("Наибольший общий делитель=%d\n",c);puts("Нажмите любую клавишу для завершения...");_getch();return 0;}Зависитmain.cppРеализуетmod.hmod.cppРисунок 4.1 – Диаграмма взаимодействия модуля и программыОглавление1024.3 * Средства создания универсальных подпрограммПри проектировании сложной программы, разработчик стремиться создать какможно более универсальные подпрограммы, чтобы сократить их количество и общийобъем программы.
Для этого С++ представляет определенные возможности. Нижерассмотрены некоторые из них.4.3.1 Параметры – многомерные массивы неопределенного размераКак было показано в разделе в С++ осуществляется контроль размеров массива, начиная со второй размерности, что ограничивает применение подпрограмм, делая их неуниверсальными, зависящими от размеров массивов.В этом случае рекомендуется использовать вспомогательные массивы указателей наодномерные массивы, которые в свою очередь могут быть массивами указателей. По каждой размерности массив является одномерным, и по правилам С++ его размерность можетбыть опущена в спецификации формальных параметров. Такой подход позволяет в телефункции обрабатывать многомерные массивы с изменяющимися размерами.Пример 4.10.
Разработать подпрограмму, независящую от размерности матрицы, которая заменяет в матрице все отрицательные элементы нулевыми.Дополним матрицу вектором указателей на ее строки (см. рисунок 4.2):float B[3][4]={1.2,-4.9 ,5.0,-8.1,-3,6.1,-8.5,9.6,3.3,-6.7,-1.2,7.8};float *ptr[]={&B[0],&B[1],&B[2]};ptrB1.2-4.95.0-8.9-36.1-8.59.63.3-6.7-1.27.8Рисунок 4.2 – Структура матрицы с дополнительным массивомАдреса начала строк, получаемые посредством оператора взятия адреса «&», не типизированы, следовательно, для сохранения этих адресов в типизированных переменныхмассива их необходимо явно преобразовать к типу float *, например:(float *)&B[0].Размерность матрицы будем передавать через специальные параметры n и m:void pereform(int n,int m, float *p[]);Полный текст программы:Оглавление103#include <locale.h>#include <stdio.h>#include <conio.h>#include "mod.h"void pereform(int n,int m,float * p[]){for(int i=0;i<n;i++)for(int j=0;j<m;j++) if (p[i][j]<0) p[i][j]=0;}int main(int argc, char* argv[]){setlocale(0,"russian");float B[3][4]={1.2,-4.9 ,5.0,-8.1,-3,6.1,-8.5,9.6,3.3,-6.7,-1.2,7.8};float *ptr[]={(float*)&B[0],(float *)&B[1],(float *)&B[2]};pereform(3,4,ptr);puts("Результирующая матрица:");for(int i=0;i<3;i++){for(int j=0;j<4;j++)printf("%5.2f",B[i][j]);printf("\n");}puts("Нажмите любую клавишу для завершения...");_getch();return 0;}Пример 4.11.