Тема: Програмування. Основні етапи розробки прикладних програм
Вид материала | Документы |
- Текст програми це набір інструкцій (команд), які можуть бути виконані комп'ютером., 221.57kb.
- Тема: Робота в середовищі програмування. Запуск програм на виконання, 202.65kb.
- 1. Мови програмування (процедурні, візуальні, специфікацій). Концепція інструментального, 757.76kb.
- Програма Комплексного вступного іспиту на окр «Спеціаліст», 814.3kb.
- Називається комплекс програмних та мовних засобів, які використовуються для створення, 149.17kb.
- План лекції Тема І основні питання. Актуальність теми Лекційний матеріал Педіатрія, 473.4kb.
- Тема. Основні етапи розв’язування прикладної задачі з використанням комп’ютера. Інформаційна, 85.92kb.
- Розробка програми (робочого плану) моніторингу світового ринку готельних та ресторанних, 33.83kb.
- Структуризація програми. Поле, метод, класс, файл, проект. Об’єктне програмування., 327.95kb.
- Рекомендації з розробки навчальних та робочих програм (зі змінами І доповненнями) львів, 500.26kb.
У двонапрямлених списках і в розглянутих раніше однонапрямлених елементи утворюють лінійну структуру даних з послідовним доступом, але на відміну від них, рух по списку можливий у двох напрямках, тому кожен елемент міститиме, окрім інформаційного поля, два вказівні поля: на наступний і попередній елементи. Список матиме два фіксованих вказівники: на перший – first; на останній – last.
Оскільки рух по списку можливий у двох напрямках, то переміщення від first до last здійснюється через вказівник next; від last до first через вказівник prev, тому формування списку за правилом стеку від first до last – це та ж сама черга від last до first і навпаки.Як і в однонапрямлених списках над двонапрямленими виконують ті ж самі операції.
Оголошення структури списку може мати вигляд:
TYPE
list2=el_list2;
el_list2=record
inf:integer;
next,prev:list2
END;
- Формування списку.
writeln('введіть кількість елементів у списку');
readln(n);
first:=nil;
last:=nil;
for i:=1 to n do
begin
writeln('введіть елемент');
readln(a);
new(p);
p.inf:=a;
if last=nil then last:=p;
p.next:=first;
first.prev:=p;
first:=p
end;
2. Перегляд. Здійснюється аналогічно, тільки може здійснюватися в обоз напрямках.
PROCEDURE RESIV;
BEGIN
p:=first;
while p.next<>nil do
begin
write(p.inf,' ');
p:=p.next
end;
writeln(p.inf,' ')
END;
3. Пошук.
p:=first;
writeln('введіть шуканий елемент');
readln(a);
i:=0;
while (p<>nil) do
begin
inc(k);
if p.inf=a then
begin
i:=i+1;
writeln('під № ',k)
end;
p:=p.next
end;
if i=0 then writeln ('немає')
else writeln ('кількість ',i)
3.Вставка перед на відмінну від однонапрямленого списку аналогічна вставці після з точністю переміни вказівників prev і next.
write('введіть елемент, після якого потрібно вставити новий');
readln(n);
write('введіть новий елемент');
readln(a);
p:=first;
while p<>nil do
begin
if p.inf=n then
begin
new(q);
q.next:=p.next;
p.next:=q;
q.prev:=p
end;
q.inf:=a;
p:=p.next
end;
5. Вилучення. Як і в попередніх випадках видалення елементу не потребує складного пошуку із порівнянням через один елемент, як у однонапрямлених списках.
write('введіть елемент, який потрібно вилучити');
readln(a);
p:=first;
while p<>nil do
begin
if p.inf=a then
begin
if p=first then
begin
first:=p.next;
p.prev:=nil
end
else
begin
if p.next<>nil then
begin
p.next.prev:=p.prev;
p.prev.next:=p.next.next
end
else
begin
p.prev.next:=nil
end;
end;
end
else
p.prev.next:=p;
p:=p.next
end;
Тема: Бінарні дерева.
Особливим видом розгалужених списків є дерева. Особливістю таких динамічних структур даних є те, що кожен елемент має декілька вказівних полів (бінарні, тернарне, п-арні дерева).
На відмінну від двонапрямлених списків, дерева не мають лінійної структури. Підлеглість елементів має розгалужену структуру – це означає, що такі списки мають один верхній елемент – голову. Він має декілька підлеглих елементів, які в свою чергу мають декілька елементів, причому на будь-якому рівні всі вказівники повинні бути зв’язані з іншими елементами, якщо ж ні, то в таких випадках вони вказують в nil.
Доступ до елементів дерева здійснюється послідовно з вершини дерева через фіксований вказівник top. Елементи дерева, обидва вказівники яких зв’язані із підлеглими елементами, називають вузловими вершинами або гілками. Елементи дерева, обидва вказівники яких вільні, називаються листками або висячими вершинами. Елементи дерева, один з вказівників яких вільний, називаються напівисячими вершинами.
Шириною бінарного дерева називається кількість вузлів від крайнього лівого до крайнього правого елементу. Висотою бінарного дерева називається максимальна кількість рівнів елементів від вершини дерева до найдальшого листка.
Бінарні дерева заповнюються не довільним чином або у порядку слідування елементів, а за алгоритмом, який формує їх у вигляді впорядкованої структури.
В якості вершини дерева завжди розміщується перший по-порядку елемент. Кожен новий елемент дерева розміщується у ньому так, щоб він опинився лівіше від поточної вершини, якщо він менший від неї; і правіше, якщо він більший від неї.
Як видно з розглянутого алгоритму, структура бінарного дерева залежить від порядку введення елементів.
Оголошуються бінарні дерева подібно до лінійних списків, у вигляді вказівного типу на елемент дерева. Елемент дерева матиме три поля: інформаційне довільного типу і два вказівних відповідно на лівий і правий елементи.
USES CRT;
TYPE
bintree=el_bintree;
el_bintree=record
left,right:bintree;
inf:integer;
END;
Дії, які можна проводити з бінарними деревами:
- Формування.
- Перегляд.
- Пошук.
- Встановлення ширини та висоти дерева.
- Встановлення кількості вузлових вершин, листків та напівисячих вершин.
- Доповнення новим елементом (операція вставки перед (чи після) не має змісту, оскільки будь-який елемент у дереві розміщується в строго визначеному йому місці).
- Вилучення елемента із дерева.
Враховуючи, що сама структура дерева має рекурсивний характер, то всі програми, що реалізують відповідні операції теж будуть рекурсивними підпрограмами. Згідно із описаним вище алгоритмом формування бінарного дерева, рекурсивна процедура, що здійснює це, може мати вигляд:
PROCEDURE ZAPUS (var p:bintree;a:integer);
BEGIN
new(p);
p.left:=nil;
p.right:=nil;
p.inf:=a;
END;
PROCEDURE ROZM(var q:bintree;a:integer);
BEGIN
if q.inf
begin
if q.left<>nil then ROZM (q.left,a)
else ZAPUS(q.left,a);
end;
if q.inf>a then
begin
if q.right<>nil then ROZM(q.right,a)
else ZAPUS(q.right,a);
end;
END;
PROCEDURE INPUT;
BEGIN
top:=nil;
writeln('введіть кількість елементів');
readln(n);
for i:=1 to n do
begin
writeln('введіть елемент:');
readln(k);
if top=nil then ZAPUS(top,k)
else ROZM(top,k);
end;
END;
Враховуючи рекурсивність структури і впорядкованість елементів, вивід їх на екран здійснюватиме теж рекурсивна процедура за порядком зростання, за алгоритмом:
- Здійснюватиметься рух по дереву до крайнього лівого елемента (найменшого).
- Це значення друкується.
- Робиться один крок вправо і послідовність 1-3 повторюється відносно нової поточної вершини.
Програмна реалізація алгоритму друку має вигляд процедури:
PROCEDURE OUTPUT (p:bintree);
BEGIN
if p.left<>nil then OUTPUT(p.left);
write(p.inf,' ');
if p.right<>nil then OUTPUT(p.right);
END;
Пошук елементу за вказаним значенням подібний до алгоритму виведення, при цьому замість процедури друку використовується порівняння інформаційних полів.
PROCEDURE FIND (p:bintree; x:integer);
BEGIN
inc(i);
if p.right<>nil then FIND(p.right,x);
if p.inf=x then
begin
write('є під номером(ами) ');
Str(i,d);
s:=s+d;
end;
if p.left<>nil then FIND(p.left,x);
if (p=top) and (s='') then writeln('елемент не знайдено');
END;
Щоб визначити ширину дерева достатньо здійснити рекурсивний рух вліво і вправо до кінця відносно вершини із підрахунком кількості кроків, а потім додати.
PROCEDURE MOVE_RIGHT(p:bintree);
BEGIN
if p.right<> nil then
begin
k:=k+1;
MOVE_RIGHT(p.right)
end;
END;
PROCEDURE MOVE_LEFT(p:bintree);
BEGIN
if p.left<> nil then
begin
k:=k+1;
MOVE_LEFT(p.left)
end;
END;
В основній програмі це буде мати вигляд:
n:=0;
k:=0;
MOVE_RIGHT(top);
MOVE_LEFT(top);
writeln('ширина :',n+k+1);
Процедура визначення висоти бінарного дерева буде мати вигляд:
PROCEDURE HIGHT(p:bintree;k:integer);
BEGIN
if p.left<>nil then HIGHT(p.left, k+1);
if k>h then h:=k;
if p.right<>nil then HIGHT(p.right,k+1)
END;
Для встановлення кількості висячих вершин використовують таку процедуру:
PROCEDURE LUSTOK(p:bintree;var l:word);
begin
if (p.left=nil) and (p.right=nil) then inc(l);
if p.left<>nil then LUSTOK(p.left,l);
if p.right<>nil then LUSTOK(p.right,l);
end;
Доповнення новим елементом проводиться так само як формування дерева.
Складність видалення пов’язана із тим, що приходиться аналізувати декілька можливих ситуацій:
а). Видалення висячої вершини найбільш проста. Вона передбачає встановлення у nil вказівника на елемент, що видаляється, для цього потрібно мати фіксований вказівник на попередній елемент.
Dispose(p.right);
p.right:=nil;
Dispose(p.left);
p.left:=nil;
б). Видалення напіввисячої вершини. При цьому всі елементи, що знаходяться нижче елемента, що видаляється підтягуються на один рівень вверх.
Тема: Модульний принцип організації програм.
Структурне програмування передбачає розділення алгоритмів на під алгоритми і програм на підпрограми. Такий поділ здійснюється не простим розділенням тексту програми на куски, а на логічно завершені цілісні компоненти. Об'єднання таких компонентів забезпечує об'єднання дій.
В ранніх компіляторах мови Turbo Pascal з'явилася перша можливість розділення програм на окремі модулі, при цьому в модулях зосереджувалися окремі компоненти мови, що пов'язувалися спільними властивостями.
Ці модулі окремо компілювалися і були доступними для всіх програм; це були свого роду бібліотеки інструментів мови програмування.
Пізніше модульний принцип став переважати у програмуванні і особливо з початком розробки складних програмних систем. Теперішні програми фактично є невеликим виконуваним файлом, який використовує багато різних бібліотек.
З розвитком мов програмування з'явилися бібліотеки динамічної компоновки.
В Pascal модулі є окремими програмними одиницями, але не самостійно виконуваними. Окремі їх компоненти використовуються головними програмами або іншими зовнішніми модулями.
Результатами компіляції є файл, компоненти якого готові до виконання. Такі файли мають розширення *.tpu (Turbo Pascal Unit). Всі модулі, що використовуються однією програмою або іншими модулями повинні бути оголошені
USES <ім'я модуля>;
При такому оголошені компілятор в процесі трансляції і компоновки програми у виконуваний модуль здійснює пошук готових відкомпільованих компонентів у цих модулях. Якщо ж ні сама програма, ні виконувані нею модулі не містять дійсних описань, тоді виникає помилка компілятора. Тема: Структура модуля.
Як вже відмічалося модуль подібний до Pascal-програми, але є певні відмінності. В модулях виділяють чотири розділи:
- Заголовок.
UNIT <ім'я модуля>;
На відмінну від програм, заголовок модуля є обов’язковим. Ім'я модуля повинно співпадати з іменем відповідного *.pas-файлу, що містить текст модуля (8 букв).
- Інтерфейс на частина.
ІNTERFACE
TYPE
compl=record
re,im:real
END;
PROCEDURE SUMA (a,b:compl; var d:compl);
PROCEDURE RIZNUTSYA (a,b:compl; var d:compl);
PROCEDURE DOBYTOK (a,b:compl; var d:compl);
PROCEDURE CHASTKA (a,b:compl; var d:compl);
FUNCTION MODYL (a:compl):real;
FUNCTION ARGYMENT (a:compl):real;
В розділі інтерфейс оголошуються всі компоненти модуля, структура яких і самі вони мають бути „видимими” зовнішнім програмам, що використовують цей модуль.
В інтерфейсі можуть оголошуватися: uses- специфікація всіх модулів, що використовуються даним модулем; константи; типи; мітки; змінні; заголовки підпрограм. Всі перелічені компоненти будуть доступними, тобто „видимими” іншим програмам.
Підпрограми зовні модуля будуть видимі лише своїм інтерфейсом, причому описується, що це за підпрограма: процедура чи функція; ім'я, повний список формальних параметрів та тип результату для функції.
Такий спосіб оголошення підпрограм приховує внутрішню реалізацію відповідних підпрограм, а залишає доступним лише перелік параметрів та ім'я.
- Розділ реалізації. В реалізації описуються всі компоненти мови, які мають бути невидимі зовні.
Тут можуть бути оголошення констант, типів змінних, підпрограм для внутрішнього використання, а також використовуються реалізації всіх підпрограм, заголовки яких містяться в інтерфейсі.
При описі реалізації підпрограм в цьому розділі може задаватися лише скорочений заголовок без списку формальних параметрів.
implementation
PROCEDURE SUMA;
BEGIN
d.Re:=a.Re+b.Re;
d.Im:=a.Im+b.Im;
END;
PROCEDURE RIZNUTSYA;
BEGIN
d.Re:=a.Re-b.Re;
d.Im:=a.Im-b.Im;
END;
- Розділ ініціалізації початкових значень змінних. Цей розділ є необов’язковим і містить оператори присвоєння змінним із розділу інтерфейс, якщо вони є деяких початкових значень, якщо вони відмінні від значень по-замовчуванню.
Якщо в інтерфейсі не оголошувалася жодна змінна або їй не потрібно надавати певного значення відмінного від замовчування, то розділ ініціалізації може бути пустим.
Розділ ініціалізації обмежується операторними дужками. Якщо модуль не містить жодної ініціалізації змінних, то слово begin може бути відсутнім.
Тема: Компіляція модулів.
Після того, як модуль повністю заповнений відповідними оголошеннями, його ще не можна використовувати іншими програмами, адже програми користуються не текстовим описанням, а вже відкомпільованими компонентами.
Компіляція модуля повинна здійснюватися не у пам'яті, а на диск у вигляді *.TPU-файла. Це можна здійснити, вибравши відповідний пункт меню інтегрованої оболонки та підпункт Destination disk (по замовчуваннюBest nation memory). Після цього компіляція буде здійснюватися на диск у вигляді типового файла.
Компільований модуль може бути підключений до будь-якої програми, при цьому він має бути в тому самому каталозі звідки запускається інтегрована оболонка turbo.exe, або в підпункті directories.
В підпункті directories має бути вказаний шлях до каталогу, де знаходиться цей модуль.
Тема: Видимість ідентифікаторів у модулях.
Часто виникає ситуація, коли деякий модуль А використовується в програмі Р, в свою чергу модуль А використовує ресурси іншого модуля В, а той в свою чергу користується модулем С, крім цього кожен з модулів С, В, А і сама програма Р можуть містити оголошення деяких навіть різних компонентів, але з одним іменем. Якщо має місце таке вкладене використання модулів, то як правильно оголосити uses- специфікацію в кожному модулі і програмі, а також як отримати доступ до однойменних об’єктів різних рівнів. Для цього потрібно користуватись такими правилами:
- Якщо програма явно користується ресурсами модуля А (не явно, тобто через ресурси модуля А) ресурсами модуля В. Модуль А явно використовує модуль В і неявно (тобто через модуль В) використовує модуль С і т. д. То в uses- специфікації потрібно вказувати всі модулі, які використовуються явно або неявно, причому у порядку, який є зворотним до порядку вкладення модулів. Таким чином матимемо оголошення:
UNIT C;
INTERFACE
... ... ...
IMPLEMENTATION
... ... ...
BEGIN
END
UNIT B; UNIT A;
INTERFACE INTERFACE
USES C; USES B,C;
... ... ...
PROGRAM P;
USES C,B,A;
VAR
xx: real;
Якщо в декількох модулях і програмах є однойменні об’єкти, то доступним вважається той з об’єктів, який, або оголошений в поточному модулі, або у першому по-порядку в uses- специфікації з право наліво, якщо у поточному модулі чи програмі його оголошення відсутнє.