Студопедия Главная Случайная страница Обратная связь

Разделы: Автомобили Астрономия Биология География Дом и сад Другие языки Другое Информатика История Культура Литература Логика Математика Медицина Металлургия Механика Образование Охрана труда Педагогика Политика Право Психология Религия Риторика Социология Спорт Строительство Технология Туризм Физика Философия Финансы Химия Черчение Экология Экономика Электроника

ПРАКТИЧЕСКАЯ ЧАСТЬ. Пример 1. Написать программу сортировки по возрастанию заданного массива случайных чисел, равномерно распределенных в интервале [–6; 6]





Пример 1. Написать программу сортировки по возрастанию заданного массива случайных чисел, равномерно распределенных в интервале [–6; 6], с помощью вспомогательной функции.

Программный код решения примера

#include <stdio.h> #include <conio.h> #include <stdlib.h> #include <time.h> #define MAX 10 // Прототип функции с формальными параметрами void sort(double arr[], int n); int main (void) { double M[MAX]; int i, size = MAX; long int L; time_t t; srand((unsigned)time(&t)); for (i = 0; i < MAX; ++i) M[i] = 12.0*rand()/RAND_MAX - 6.0; printf("\n\t The original array:\n"); for (i = 0; i < MAX; ++i) {printf("\t%8.4f\n", M[i]);} sort(M, size);// обращение к функции с фактическими параметрами // Распечатка отсортированного массива printf("\n\t After sorting: \n"); for (i = 0; i < MAX; ++i) printf("\t%8.4f\n", M[i]); printf("\n Press any key: "); getch(); return 0; } // Пользовательская функция сортировки void sort(double Array[], int m) { int i, j; double tmp; for (i = 0; i < m-1; ++i) for (j = 0; j < m-i-1; ++j) if (Array[j+1] < Array[j]) { tmp = Array[j]; Array[j] = Array[j+1]; Array[j+1] = tmp; } }

Следует обратить внимание на имена формальных параметров в самой функции sort() и в ее прототипе: они имеют разные имена, но одинаковые типы. Фактические параметры, или аргументы, функции sort() в вызывающей программе (в теле функции main()) имеют свои имена, не связанные с именами формальных параметров.

Заполнение массива случайными числами производится с помощью библиотечной функции rand() и макроопределения RAND_MAX.


Возможный результат выполнения программы представлен на рис. 10.2.

Рис. 10.2. Пример сортировки числового массива

Задание 1

1. Используйте программу без прототипа функции sort().

2. С помощью вспомогательной функции произведите сортировку массива по убыванию.

3. С помощью вспомогательной функции произведите сортировку целых чисел из заданного интервала [X; 10X], где Х – номер компьютера, на котором выполняется лабораторная работа.

4. Создайте функцию умножения исходного массива как матрицы-столбца на отсортированный массив как матрицу-строку. Выведите на консоль результат перемножения.

5. Создайте функцию умножения исходного массива как матрицы-строки на отсортированный массив как матрицу-столбец. Выведите на консоль результат перемножения.

Пример 2. Написать программу вычисления квадратного корня числа по методу Ньютона – Рафсона с использованием функции расчета квадратного корня числа и функции расчета абсолютного значения числа.

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

Алгоритм метода Ньютона – Рафсона для вычисления квадратного корня числа.

Шаг 1. Выбрать приблизительное значение 1.

Шаг 2. Если | guess 2x | < e, перейти к шагу 4 (англ. guess – предположение).

Шаг 3. Установить приблизительное значение, равное (x / guess + guess)/2, и перейти к шагу 2.

Шаг 4. Считать приблизительное значение квадратным корнем числа [5].

Программный код решения примера

#include <stdio.h> #include <conio.h> // Функция абсолютного значения числа double absValue(double x) { if (x < 0) x = -x; return (x); } // Функция расчета квадратного корня из числа double squareRoot (double x) { const double epsilon = 0.000001; double guess = 1.0; //Начальное приближение if (absValue(guess*guess - x) >= epsilon) guess = (x/guess + guess)/2.0; return (guess); } int main (void) { double result, X; printf("\n\t Enter a number: "); scanf_s("%lf", &X); // Обращение к функции с фактическим параметром result = squareRoot(X); printf("\n\t Square root of \"%1.4f\" is: %1.8f\n", X, result); printf("\n Press any key: "); getch(); return 0; }

В программе используются три функции: main(), absValue(), squareRoot().

Вспомогательные функции расположены в определенном порядке: сначала absValue(), а потом squareRoot(). В первую передается значение guess, которое вычисляется в функции squareRoot(), находящейся в условии оператора цикла if. Когда условие для оператора цикла будет ложным, т. е. когда значение корня будет меньше заданного числа epsilon, полученное значение возвращается в вызывающую функцию main().

В программе использована переменная epsilon, определенная с помощью спецификатора const, что делает ее неизменной. При таком объявлении компилятор определяет ее как константное значение. Таким переменным нельзя присваивать значения в программе, их нельзя инкрементировать или декрементировать [2].


Возможный результат выполнения программы показан на рис. 10.3.

Рис. 10.3. Результат итерационного извлечения квадратного корня из двух

Задание 2

1. В функции absValue() примените оператор условия? вместо оператора if без определения дополнительной переменной.

2. В главной функции main() введите проверку ввода неотрицательного числа.

3. В программу введите прототипы функций.

4. Зафиксируйте результат вычисления квадратного корня от числа Х, где Х – номер компьютера, на котором выполняется лабораторная работа.

5. В качестве начального приближения возьмите 1,5Х, где Х – номер компьютера, на котором выполняется лабораторная работа.

6. В программе примените тип float вместо типа double.

Пример 3. Написать программу поиска максимального элемента среди минимальных элементов строк двухмерного целочисленного массива.

Условие примера соответствует поиску максмина в двухмерном массиве, т. е.

где – элементы матрицы А.

Алгоритм поиска максмина заключается в следующем [6]. Сначала предполагаем, что максимальным элементом является начальный элемент первой строки A[0][0], затем заменяем его минимальным элементом этой же строки, т. е. теперь минимальный элемент первой строки принимается за искомый максимум. Последовательно просматривая остальные строки, находим в каждой из них минимальный элемент. Если окажется, что минимальный элемент текущей строки больше текущего максимума, то он принимается за максимальный. Результатом поиска являются значение максимального элемента и его индексы (номер строки и номер столбца).

Программный код решения примера

#include <stdio.h> #include <conio.h> #include <stdlib.h> #include <time.h> #define n 6 #define m 7 const int N = 100; int main (void) { //Прототип функции int MaxMin(int A[][m], int nn, int mm, int *imax, int *jmax); int i, j, A[n][m], max, imax, jmax; time_t t; srand((unsigned)time(&t)); //Заполнение матрицы целыми случайными числами из [-N; N-1] for (i = 0; i < n; ++i) for (j = 0; j < m; ++j) A[i][j] = (rand() % N) - N; // Распечатка матрицы printf("\n\t The original matrix A(%d*%d):\n\n", n, m); for (i = 0; i < n; ++i){ printf("\t"); for (j = 0; j < m; ++j) printf("%4d", A[i][j]); printf("\n"); } max = MaxMin(A, n, m, &imax, &jmax); printf("\n\t Result: MaxMin = A[%d][%d] = d\n",imax+1,jmax+1, max); printf("\n Press any key: "); getch(); return 0; } // Функция поиска максмина int MaxMin(int A[][m], int nn, int mm, int *imax, int *jmax) { int i, j, min, max, imin, jmin; max = A[0][0]; *imax = 0; *jmax = 0; for (j = 1; j < mm; ++j) if (A[0][j] < max) {max = A[0][j]; *imax = 0; *jmax = j;} for (i = 1; i < nn; ++i) { min = A[i][0]; imin = i; jmin = 0; for (j = 1; j < mm; ++j) if (A[i][j] < min) {min = A[i][j]; imin = i; jmin = j;} if (max < min) {max = min; *imax = imin; *jmax = jmin;} } return max; }


В объявлении функции MaxMin() для матрицы явно указывается только второй размер – количество столбцов. Указатели *imax, *jmax используются для передачи в вызывающую функцию индексов найденного максмина. Возвращение самого максмина осуществляется по значению с помощью оператора return, т. е. return max..

Возможный результат выполнения программы представлен на рис. 10.4.

Рис. 10.4. Пример вычисления максмина двухмерного массива

Задание 3

1. Подсчитайте число итераций, используемых при поиске максмина.

2. В объявлении матрицы MaxMin() (и ее прототипа) примените указатель на одномерный массив, например int *a вместо int A[ ][m].

3. Число строк матрицы задайте как 3Х, где Х – номер компьютера, на котором выполняется лабораторная работа.

4. Размерность анализируемого массива (матрицы) задайте с клавиатуры. С клавиатуры задайте также интервал генерирования случайных чисел функцией rand() в виде [–2Х; 2Х], где Х – номер компьютера, на котором выполняется лабораторная работа.

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

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

Пример 4. Написать программу поиска одного элемента в линейной неупорядоченной таблице по совпадению ключа на основе заграждающего элемента. Таблицу описать в виде одномерного массива целых чисел [6].

В линейной таблице элементы располагаются друг за другом, т. е. для каждого из них существуют отношения порядка [6]. Линейные таблицы в оперативной памяти компьютера представляются массивами или линейными связанными списками.

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

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

Программный код решения примера

#include <stdio.h> #include <conio.h> #define size 10 // Таблица числовых элементов int A[size] = {1,-7,3,4,8,-5,-2,6,0,9}; int main (void) { int i, ind, key; int search(int A[], int n, int key);// прототип функции printf("\n\t The original array:\n\t"); for (i = 0; i < size; ++i) printf("%3d", A[i]); // Поиск ключа соответствия printf("\n\n\t Search for key matches.\n"); // Ввод ключа поиска printf("\t Key in your search: "); scanf_s("%i", &key); ind = search(A, size, key); // обращение к функции if (ind == -1) printf("\n\t Element \"%d\" could not be found\n", key); else printf("\n\t Element \"%d\" index is %d\n", key, ind+1); printf("\n Press any key: "); getch(); return 0; } int search(int A[], int n, int key) { int i = 0, r; r = A[n-1]; A[n-1] = key;// заграждающий элемент if (A[i]!= key) i++; A[n-1] = r; // восстановление последнего элемента if ((i == n - 1) && (r!= key)) return -1; // отсутствие элемента else return i; // успешный поиск }

В программе использовано внешнее объявление одномерного массива с инициализацией. Термин «внешний» здесь используется, чтобы подчеркнуть расположение объявления вне функций; напрямую с ключевым словом extern он не связан [4].


Возможный результат выполнения программы показан на рис. 10.5.

Рис. 10.5. Пример поиска элемента в массиве по заданному ключу

Задание 4

1. Подсчитайте число итераций при поиске элемента по заданному ключу.

2. Напишите программу без заграждающего элемента. Подсчитайте число итераций при поиске элемента по заданному ключу. Сравните с числом итераций при поиске с заграждающим элементом.

3. Сформируйте динамический одномерный массив размерностью 12Х, и заполните его целыми случайными числами, распределенными равномерно из интервала [0; 20X], где Х – номер компьютера, на котором выполняется лабораторная работа. Ввод ключа для поиска выполните с клавиатуры. Поиск выполните с заграждающим элементом.

4. С учетом предыдущего пункта задания ввод ключа задайте случайным образом. Если по этому ключу элемент не найден, то программным путем организуйте повторный поиск с заданием нового случайного ключа. В случае если при 5Х-кратном поиске элемент не найден, выведите сообщение об отсутствии искомого элемента по всем заданным ключам. Поиск выполнить с заграждающим элементом.

Пример 5. Написать программу выделения слов из символьной строки, когда слова в ней разделены пробелами, и поместить каждое слово в отдельной строке свободного (вспомогательного массива) [6].

Программный код решения примера

#define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <conio.h> #include <stdlib.h> #include <string.h> #define N 123 #define M 40 int wordstr(char**c, char***FA) { int Ln, n = 0, i = 0; char**wr; char**tempstr; Ln = strlen(c); tempstr = (char**)calloc(Ln + 1, sizeof(char)); strcpy(tempstr, c); wr = strtok(tempstr, " "); Ln = strlen(wr); FA[i] = (char**)calloc(Ln + 1, sizeof(char)); strcpy(FA[i++], wr); n++; if (wr = strtok(NULL, " ")) { Ln = strlen(wr); FA[i] = (char**) calloc(Ln + 1, sizeof(char)); strcpy(FA[i++], wr); n++; } free(tempstr); return n; } int main (void) { int i, n; char**Farr[M], str[N]; printf("\n\t Enter the string of characters:\n >> "); gets_s(str, N-1); printf("\n\t The original string of characters:\n"); printf(" %c\n", str); //Обращение к функции обработки строки n = wordstr(str, Farr); //Распечатка слов, помещенных в свободный массив printf("\n\t The characters in a free array:"); for (i = 0; i < n; ++i) printf("\n\t %c", Farr[i]); //Освобождение памяти, занятую выделенными словами for (i = 0; i < n; ++i) {free(Farr[i]); Farr[i] = NULL;} printf("\n\n Press any key: "); getch(); return 0; }

В программе использована библиотечная функция strtok(), которая выделяет слова из строки. При этом для каждого выделенного слова получаем динамическую память с помощью функции calloc(). Адрес выделенной памяти помещаем в соответствующий указатель массива свободных строк FA[]. Обратите внимание, что в функции calloc() вводится число на единицу больше, чем длина строки. Это сделано для учета символа завершения строки (или одного слова) '\0'.

Для исключения предупреждений о безопасной работе с функциями strcpy(), strtok() в MS Visual Studio в программе используется директива #define _CRT_SECURE_NO_WARNINGS.


Результат выполнения программы представлен на рис. 10.6.

Рис. 10.6. Пример выделения слов из строки


Задание 5

1. В программу включите обработку символов: запятая ',', точка с запятой ';' точки '. ' в качестве разделителей слов.

2. Используйте символ пробела для разделения слов «фамилия, имя, год поступления в вуз».

3. В программу включите прототип функции wordstr().

4. Изучите возможность применения в программе функций strcpy_s() и strtok_s().

Пример 6. Написать программу расчета суммы и среднего арифметического числовых произвольного количества данных. В качестве ключевых слов для выбора варианта расчета принять mean (среднее) и sum (сумма).

В данном примере необходимо использоватеь функцию с переменным числом агрументов. Формула расчета среднего арифметического (m):

 

 

где n – количество числовых данных, xi – текущее число.

Программный код решения примера

#include <stdio.h> #include <conio.h> #include <string.h> #include <stdarg.h> #include <float.h> /* Прототип функции с переменным числом аргументов*/ double varfun(char str[], double v1,...); int main(void) { double v1 = 10.0, v2 = 2.5, control; char str[121]; printf("\n Enter one of the key words 'mean' or 'sum': "); gets_s(str, 120); control = varfun(str, v1, v2, 7.5, 0.0); if (control < DBL_MAX) printf("\n Result: %lf\n", control); else printf("\n Invalid input keyword.\n"); printf("\n\n... Press any key: "); getch(); return 0; } /* Определение функции с переменным числом аргументов */ double varfun(char str[], double v1,...) { /* Указатель на переменные списка аргументов */ va_list parg; double sum = v1; double value = 0.0; /* значение аргумента */ int count = 1; /* начальное количество аргументов */ int k, p; char**ch = "mean"; char**ch2 = "sum"; // Лексиграфическое сравнение строк k = strcmp(str, ch); if (!k) // k == 0 p = 1; else if (k) // k!= 0 { k = strcmp(str, ch2); if (!k) p = 2; else p = 0; } if (p == 1) { va_start(parg,v1); /* инициализация указателя parg */ /* Просмотр списка аргументов*/ if((value = va_arg(parg, double))!= 0.0) { // Суммирование числовых аргументов функции varfun() sum += value; count++; } /* Завершение процесса счтывания аргументов */ va_end(parg); return sum/count; } if (p == 2) { va_start(parg,v1); if((value = va_arg(parg, double))!= 0.0) { sum += value; count++; } va_end(parg); return sum; } return DBL_MAX; }


Возможный вариант выполнения программы показан на рис. 10.7.

Рис. 10.7. Пример выполнения программы

Задание 6

1. Напишите функцию с явным указанием количества аргументов.

2. Включите в программу расчет исправленной выборочной дисперсии (D), которая рассчитывается по формуле

 

где m – среднее арифметическое заданных чисел.

 

 

Контрольные вопросы

1. Что лежит в основе всех программ, созданных на языке С?

2. Какие типы данных функция возвращать может и что не может?

3. Что такое прототип функции? Какие элементы объявления функции входят в ее прототип?

4. В чем разница между фактическими и формальными параметрами функции?

5. Какой способ передачи параметров в функциях предусматривает синтаксис языка С?

6. Как можно изменить значение аргумента функции в теле самой функции?

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

8. Можно ли использовать функцию без параметров и без служебного слова void? К каким последствиям это может привести?

9. Какие обязательные атрибуты должна содержать функция с переменным числом аргументов?

 







Дата добавления: 2015-09-06; просмотров: 1236. Нарушение авторских прав; Мы поможем в написании вашей работы!




Расчетные и графические задания Равновесный объем - это объем, определяемый равенством спроса и предложения...


Кардиналистский и ординалистский подходы Кардиналистский (количественный подход) к анализу полезности основан на представлении о возможности измерения различных благ в условных единицах полезности...


Обзор компонентов Multisim Компоненты – это основа любой схемы, это все элементы, из которых она состоит. Multisim оперирует с двумя категориями...


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

Случайной величины Плотностью распределения вероятностей непрерывной случайной величины Х называют функцию f(x) – первую производную от функции распределения F(x): Понятие плотность распределения вероятностей случайной величины Х для дискретной величины неприменима...

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

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

РЕВМАТИЧЕСКИЕ БОЛЕЗНИ Ревматические болезни(или диффузные болезни соединительно ткани(ДБСТ))— это группа заболеваний, характеризующихся первичным системным поражением соединительной ткани в связи с нарушением иммунного гомеостаза...

Решение Постоянные издержки (FC) не зависят от изменения объёма производства, существуют постоянно...

ТРАНСПОРТНАЯ ИММОБИЛИЗАЦИЯ   Под транспортной иммобилизацией понимают мероприятия, направленные на обеспечение покоя в поврежденном участке тела и близлежащих к нему суставах на период перевозки пострадавшего в лечебное учреждение...

Studopedia.info - Студопедия - 2014-2025 год . (0.01 сек.) русская версия | украинская версия