Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:

Креневич А.П.,Обвінцев О.В.C в задачах і прикладах

..pdf
Скачиваний:
86
Добавлен:
07.03.2016
Размер:
1.37 Mб
Скачать

Розділ 7. Файли

141

7.2. Текстові файли

Текстові файли у мові С описують так само, як двійкові файли. Указати, що файл текстовий можна під час відкриття файла за допомогою інструкції fopen:

fp = fopen(file_name, mode);

де file_name – ім'я текстового файла, а рядок mode для текстових файлів може мати одне зі значень:

"r", "w", "a",

де r – означає відкриття існуючого файла для читання; w – створення файла для запису; a – відкриття існуючого файла для доповнення. Наприклад,

fp = fopen("rez.txt", "w"); fp = fopen("mod.txt", "r");

Для текстових файлів залишаються також визначеними підпрограми fclose і feof.

Читання текстової інформації із текстового файла здійснюється за допомогою підпрограм:

c = fgetc(fp);

прочитати з файла fp поточний символ у змінну c. fgets(str, n, fp);

читати з файла fp не більше n-1 символів і присвоїти результат рядку str. Якщо до кінця рядка файла залишилось менше n-1 символів, то буде прочитано тільки символи до кінця рядка та символи кінця рядка.

Запис у текстовий файл текстової інформації здійснюється підпрограмами

fputc(c, fp); fputs(str, fp);

Перша із них записує у файл fp символ c, друга – рядок str.

Для читання/запису з файла/у файл інформації, що є не обов'язково текстовою, використовують функції fscanf/fprintf:

fscanf(fp, "format", variables); fprintf(fp, "format", variables);

Ці функції є аналогами інструкцій scanf/printf. Правила використання також аналогічні до scanf/printf (див. ст. 6). Єдина відмінність полягає у тому, що інструкція scanf зчитує змінні (з переліку) з клавіатури, а інструкція fscanf – з файла fp. Відповідно printf виводить змінні (з переліку) на екран, а fprintf – у файл fp.

142

С у задачах і прикладах

 

Приклади розв'язання задач

Приклад 7.5. Замінити у текстовому файлі всі символи c рядком a. Розв'язок. У програмі визначимо константи MAXLEN і MAXREPL. Пер-

ша з них відповідає за максимальну довжину рядків файла, друга – за максимальну довжину рядка a. Глобальні змінні fname і gname призначено для імен вхідного і результуючого файлів.

#include <stdio.h> #include <string.h> #include <iostream.h> #define MAXLEN 256 #define MAXREPL 10

char fname[81], gname[81];

/* Підпрограма, що замінює у рядку s всі символи c рядком a і результат записує у рядок r */

void replace(char s[],char a[],char c,char r[]){ unsigned i;

strcpy(r, "");

for (i=0; i < strlen(s); i++) if (s[i] == c) strcat(r,a); else {

r[strlen(r) + 1] = '\0'; r[strlen(r)] = s[i];

}

}

/* Підпрограма, що замінює у вхідному файлі всі символи c рядком a і результат записує у результуючий файл */

void replace_text(char c, char a[]){ FILE *fp, *gp;

char s[MAXLEN], s1[MAXLEN*MAXREPL]; fp = fopen(fname, "r");

gp = fopen(gname, "w"); while(1) {

fgets(s,MAXLEN,fp); if (feof(fp)) break; replace(s,a,c,s1); fputs(s1,gp);

}

fclose(fp); fclose(gp);

}

void main(){

char c, a[MAXREPL];

cout << "Ім'я вхідного файла: ";

Розділ 7. Файли

143

gets(fname);

cout << "Ім'я результуючого файла: "; gets(gname);

cout << "Рядок заміни: "; gets(a);

cout << "Символ для заміни: "; c = getchar(); cout << "\n"; replace_text(c,a);

}

Задачі для самостійної роботи

7.39. Дано текстовий файл. Групи символів, що відокремлені пропусками (одним або кількома) і не містять пропусків усередині, називатимемо словами. Скласти підпрограми для:

а) знаходження найдовшого слова у файлі; б) визначення кількості слів у файлі;

в) вилучення з файла зайвих пропусків і всіх слів, що складаються з однієї літери;

г) вилучення всіх пропусків на початку рядків, у кінці рядків і між словами (крім одного);

ґ) вставки пропусків до рядків рівномірно між словами так, щоб довжина всіх рядків (якщо в них більше 1 слова) була 80 символів і кількість пропусків між словами в одному рядку відрізнялась не більш ніж на 1 (вважати, що рядки файла мають не більш ніж 80 символів).

Результат записати до файла H.

7.40. Визначити процедуру, яка утворює текстовий файл із 9 рядків, у першому з яких одна літера '1', у другому – дві літери '2', ... , у де- в'ятому – дев'ять літер '9'.

7.41.Визначити процедуру, яка за заданою послідовністю символів формує текстовий файл із рядками по 40 літер (в останньому рядку літер може бути й менше).

7.42.Визначити функцію, яка:

а) підраховує кількість порожніх рядків; б) обчислює максимальну довжину рядків текстового файла. 7.43. Визначити процедуру виведення:

а) усіх рядків текстового файла; б) рядків, які містять більше 60 символів.

7.44. Визначити функцію, що визначає кількість рядків текстового файла: а) починаються із заданого символу; б) закінчуються заданим символом;

в) починаються й закінчуються одним і тим самим символом; г) що складаються з однакових символів.

144

С у задачах і прикладах

 

7.45.Визначити процедуру, яка переписує до текстового файла G усі рядки текстового файла F:

а) із заміною в них символа '0' на '1', і навпаки; б) в інвертованому вигляді.

7.46.Визначити процедуру пошуку найдовшого рядка в текстовому файлі. Якщо таких рядків кілька, знайти перший із них.

7.47.Визначити процедуру, яка переписує компоненти текстового

файла F до файла G, вставляючи до початку кожного рядка один символ пропуску. Порядок компонент не має змінюватися.

7.48.Визначити процедуру, яка друкує за рядками зміст текстового файла, вставляючи до початку кожного рядка його порядковий номер і символ пропуску.

7.49.Визначити процедуру пошуку в текстовому файлі рядків, фрагментом яких є заданий рядок.

7.50.Вважаючи, що довжина рядка текстового файла F не переви-

щує 80, визначити процедуру, яка, доповнюючи короткі рядки файла F пропусками праворуч, формує текстовий файл G, усі рядки якого мають довжину 80.

7.51.Визначити процедуру, яка, виключаючи пропуски, що стоять на початку рядків текстового файла F, формує текстовий файл G.

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

7.53.У текстовому файлі F записано послідовність цілих чисел, які розділяються пропусками. Визначити процедуру запису до текстового файла g усіх додатних чисел із F.

7.54.У текстовому файлі кожний рядок містить кілька натуральних чисел, які розділяються пропусками. Числа визначають вигляд геометричної фігури (номер) та її розміри. Прийнято такі домовленості:

відрізок прямої задається координатами своїх кінців і має номер 1;

прямокутник задається координатами верхнього лівого й нижнього правого кутів і має номер 2;

коло задається координатами центра й радіусом і має номер 3. Визначити процедури обчислення:

а) відрізка з найбільшою довжиною; б) прямокутника з найбільшим периметром; в) кола з найменшою площею.

7.55. Скласти програму для перевірки правильності розстановки фі-

гурних дужок у текстовому файлі, що є файлом програми, написаної С.

Розділ 8 СОРТУВАННЯ

ТА ШВИДКИЙ ПОШУК

У практиці програмування дуже часто виникає необхідність упорядкування елементів структур даних за деякою ознакою, що називають "ключем". У цьому випадку кажуть, що необхідно відсортувати множину елементів, використовуючи задане відношення порядку.

Означення 8.1. Сортуванням називають операцію впорядковування елементів деякої структури даних, на якій визначено відношення порядку, за ключами.

Залежно від того, чи містяться елементи структур даних у внутрішній (оперативній) або у зовнішній (на зовнішніх пристроях) пам'яті, розріз-

няють внутрішнє та зовнішнє сортування.

Найбільш поширені алгоритми внутрішнього сортування – це:

сортування включенням або вставкою;

сортування вибором;

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

сортування злиттям;

швидке сортування з використанням рекурсії.

Серед алгоритмів зовнішнього сортування найуживанішим є алгоритм сортування злиттям з використанням двох-трьох допоміжних файлів.

Інший клас важливих алгоритмів, що використовуються на практиці,

– це алгоритми, які реалізують різні методи пошуку заданого елемента серед множини компонент визначеної структури даних. Раніше ми розглядали лінійний пошук, що не накладає жодних умов на елементи, серед яких здійснюється пошук, за винятком існування механізму послідовного доступу до них. Саме це робить лінійних пошук непридатним для використання з великими обсягами даних.

Додаткову інформацію про механізми сортування і пошуку можна отримати з додаткової літератури, наприклад, [12].

Розглянемо на прикладах основні алгоритми сортування масивів і файлів, а також механізм швидкого бінарного пошуку.

Зауваження. Нагадаємо, що нумерація елементів масиву в С завжди починається з 0.

146

С у задачах і прикладах

 

Приклади розв'язання задач

Зауваження. Тут і далі при написанні підпрограм вважатимемо, що масив містить N елементів, і кожна програма містить рядок макрозаміни на кшталт

#define N 5

Приклад 8.1. Дано масив a цілих чисел, упорядкованих за зростанням. Серед елементів цього масиву знайти число x.

Розв'язок. Оскільки елементи масиву впорядковані, то для пошуку необхідного елемента можна скористатися, наприклад алгоритмом бінарного пошуку. Ідея алгоритму: на кожному кроці шукаємо елемент x серед елементів масиву (a0 , a1 ,..., aN −1 ) від верхньої до нижньої

границі. Спочатку нижня границя індексів – 0 (перший елемент масиву), а верхня N – 1 (останній елемент).

На кожному кроці коло елементів, серед яких шукаємо x, зменшуємо вдвічі. Для цього порівнюємо x із середнім елементом частини масиву від нижньої до верхньої границі та змінюємо значення нижньої або верхньої границі індексу. Отже, підпрограма бінарного пошуку матиме вигляд:

int BinarySearch(int a[], int x){ int l, r, m;

l = 0; r = N - 1; while (l < r) {

m = (r + l) / 2; // вибираємо середній елемент

if (a[m] < x)

 

l = m + 1;

// змінюємо нижню границю

else

 

r = m;

// змінюємо верхню границю

}

 

if (a[l] == x)

 

return l;

 

else

 

return -1;

 

}

 

Приклад 8.2. Упорядкувати методом включення масив цілих чисел за зростанням елементів.

Розв'язок. Сортування включенням (вставкою) передбачає

(N – 1) кроків. Алгоритм сортування:

а) припускаємо, що для деякого i, починаючи з другого елемента (i = 1), у послідовності елементів (a0 , a1 ,..., ai−1 , ai ,..., aN −1 ) підпослідов-

ність (a0 , a1,..., ai−1 ) вже відсортовано;

б) вибираємо i-й елемент ai і переносимо його до вже відсортованої послідовності (a0 , a1,..., ai−1 ), вставляючи його на своє місце.

Розділ 8. Сортування та швидкий пошук

147

 

в) проробляємо вищеописані кроки а) та б) для всіх i від 1 до N – 1 . Залежно від того, як відбувається включення елемента аі до підпос-

лідовності (a0 , a1,..., ai−1 ), розрізняють пряме та бінарне включення.

При сортуванні прямим включенням елемент аі послідовно порівню-

ється з елементами

(a0 , a1 ,..., ai −1 ) доки не буде знайдено такий еле-

мент a j , що a j ≤ ai

( j = 0, ..., i −1 ). При цьому всі елементи масиву

(від j + 1 до i – 1 ) зсуваються на одну позицію праворуч, а на j+1 позицію вставляється елемент аі.

Наступна підпрограма StraightInsertionSort здійснює сортування масиву a методом прямого включення:

void StraightInsertionSort(int a[]){ int i, j, x;

for (i = 1; i <= N-1; i++){ x = a[i];

j = i-1;

// наступний цикл шукає елемент a[j] <= a[i] while (j >= 0){

if (a[j] <= x)

break; // a[j] знайдено – припиняємо пошук else

a[j+1] = a[j]; // зсуваємо елементи масиву j = j--;

}

a[j+1] = x; // на j+1-шу позицію ставимо a[i]

}

При сортуванні бінарним включенням для пошуку елемента аj серед елементів (a0 , a1,..., ai−1 ) такого, що a j ≤ ai використовується бінарний

пошук.

Підпрограма BinaryInsertionSort здійснює сортування масиву a методом бінарного включення:

void BinaryInsertionSort(int a[]){ int i, j, l, r, m, x;

for (i = 1; i <= N-1; i++){ x = a[i];

l = 0; r = i;

// здійснюємо бінарний пошук while (l < r){

m = (l + r) / 2; if (a[m] <= x)

l = m + 1; else

r = m;

}

148

С у задачах і прикладах

 

//зсув елементів масиву на один вправо for (j = i; j >= r+1; j--)

a[j] = a[j-1];

//вставка a[i] у потрібну позицію

a[r] = x;

}

}

Приклад 8.3. Методом сортування вибором впорядкувати масив цілих чисел за зростанням елементів.

Розв'язок. При сортуванні вибором на кожному кроці для всіх i

від 0 (першого елемента) до N – 2 (передостаннього) знаходимо елемент аj – найменший серед елементів підпослідовності (ai ,..., aN 1 ).

Після цього міняємо місцями елементи аi та аj.

Підпрограма SelectionSort здійснює сортування масиву a вибором:

void SelectionSort(int a[]){ int i, j, k, x;

for (i = 0; i <= N - 2; i++){

//пошук мінімального серед a[i],... ,a[N-1] j = i;

for (k = i+1; k <= N-1; k++) if (a[k] <= a[j])

j = k;

//міняємо місцями a[i] i a[j]

x = a[j]; a[j] = a[i]; a[i] = x;

}

}

Приклад 8.4. Упорядкувати масив цілих чисел, використовуючи обмінне сортування.

Розв'язок. Алгоритм обмінного сортування базується на порів-

нянні пар сусідніх елементів масиву. За необхідності елементи у парі міняються місцями. Так продовжується до впорядкування всіх елементів. Цей метод також відомий як метод "бульбашки", оскільки за один крок зовнішнього циклу найменший ("найлегший") елемент серед розглянутих елементів масиву переміщується до початку масиву (немов би "спливає" нагору).

void BubbleSort(int a[]){ int i, j, x;

for (i = 1; i <= N-1; i++) for (j = N-1; j >= i; j--)

if (a[j-1] > a[j]){

x = a[j-1]; a[j-1] = a[j]; a[j] = x;

}

Розділ 8. Сортування та швидкий пошук

149

 

}

Приклад 8.5. Використовуючи сортування злиттям, упорядкувати масив цілих чисел.

Розв'язок. Для початку розглянемо задачу злиття двох упорядкованих масивів a = (a0 , a1 ,..., an ) та b = (b0 ,b1 ,...,bm ) у впорядкований ма-

сив c =(c0 , c1 ,...,cn+m ), що містить усі елементи масивів a та b. Цю

задачу розв'язує фрагмент програми: i = j = k = 0;

while (i <= n && j <= m) { if (a[i] < b[j]){

c[k] = a[i]; i++;

}

else {

c[k] = b[j]; j++;

}

k++;

}

while (i <= n) { c[k] = a[i]; i++; k++;

}

while (j <= m) { c[k] = b[j]; j++; k++;

}

Для кожного конкретного випадку другий або третій цикл while не виконується жодного разу, залежно від того, який масив вичерпується раніше a чи b.

При сортуванні злиттям на i-му кроці масив a розбивається на впорядковані підмасиви (послідовності елементів масиву, що йдуть поспіль) довжини l = 2i−1 . Пари сусідніх підмасивів зливаються у впорядковані підмасиви довжиною 2l у допоміжному масиві b. Після присвоєння масиву a значення масиву b і збільшення l удвічі, злиття повторюється для підмасивів більшого розміру. Процес завершується, коли l стає більшим або рівним N – 1 . Оскільки при i = 1 підмасиви, що складаються з одного елемента, упорядковані за означенням, то у результаті отримуємо впорядкований масив a.

Для реалізації цього алгоритму опишемо допоміжну підпрограму Merge, яка зливає пари впорядкованих підмасивів довжини Size масиву a та результат записує у масив b. У ній r1 – права границя індексів першого підмасиву пари, r2 – права границя індексів другого підмасиву пари. У підпрограмі врахуємо, що, оскільки Size набуває значення

150

С у задачах і прикладах

 

степені 2, а N - 1 може не бути кратним степені 2, то довжина підмасивів останньої пари може бути меншою за Size.

void Merge(int a[], int Size, int b[]){ int i, j, k, r1, r2;

k = 0;

while (k <= N-1){

//визначення границь підмасивів i = k;

r1 = i + Size - 1;

if (r1 > N-1) r1 = N-1; j = r1 + 1;

r2 = j + Size - 1;

if (r2 > N-1) r2 = N-1; // злиття пари підмасивів

while (i <= r1 && j <= r2) { if (a[i] < a[j]){

b[k] = a[i]; i++;

}

else {

b[k] = a[j]; j++;

}

k++;

}

while (i <= r1) { b[k] = a[i]; i++; k++;

}

while (j <= r2) { b[k] = a[j]; j++; k++;

}

}

}

Оскільки у С немає механізмів копіювання масивів як цілісних структур, то опишемо допоміжну підпрограму ArrayCopy, що копіює масив b у масив a.

void ArrayCopy(int a[], int b[]){ for (int j = 0; j <= N-1; j++)

a[j] = b[j];

}

Використовуючи вищенаведені допоміжні підпрограми, можна записати підпрограму сортування у простому вигляді:

void MergeSort(int a[]){

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]