Динамічні структури даних

Содержание

Слайд 2

Динамічні структури даних

Довільна програма створюється для обробки даних, від способу організації

Динамічні структури даних Довільна програма створюється для обробки даних, від способу організації
яких суттєво залежать алгоритми її роботи, тому вибір структур даних має першочергове значення.
Розглядали “стандартні” способи організації даних, що основані на використанні скалярних та структурних типів. Спільне – структури мали фіксований розмір на протязі роботи з ними (масиви, структури, масиви структур, …). Це визначає суттєві обмеження.
Але пам`ять під данні може виділятися або на етапі компіляції (при цьому її розмір фіксується), або у ході виконання програми – областями вказаного розміру (розмір може змінюватись за потребою).

Слайд 3

Динамічні структури даних

Виділення пам`яті під час виконання програми надає гнучкість у представлення

Динамічні структури даних Виділення пам`яті під час виконання програми надає гнучкість у
даних. Пам`ять виділяється, та звільняється блоками (неперервними), що зв`язані за допомогою покажчиків. Такий спосіб організації даних називають динамічними структурами даних, оскільки їх розмір може змінюватись у ході роботи.
Використовують як лінійні, так й нелінійні динамічні структури:
лінійні списки (стеки, черги);
дерева (бінарні, двійкового пошуку, збалансовані, …).
Вказані структури розрізняються як способами зв`язку між окремими елементами структури, так й доступними операціями.

Слайд 4

Динамічні структури даних

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

Динамічні структури даних Динамічні структури дозволяють гнучко й ефективно працювати з даними
яких заздалегідь невідомий, а також пришвидшити роботу з даними при виконанні операцій додавання, вилучення, пошуку, впорядкування даних.
Існуючі можливості для ефективного представлення та обробки різноманітної інформації є предметом подальшого розгляду.
Для динамічних структур основні:
базові типи даних – структури, покажчики, масиви;
операції – виділення та звільнення пам`яті, доступу до даних через покажчики.

Слайд 5

Інструментарій С++

Елемент динамічної структури представляється структурою, що містить принаймні два поля:

Інструментарій С++ Елемент динамічної структури представляється структурою, що містить принаймні два поля:
для збереження даних (довільні типи та кількість полів), для збереження зв'язків (поля покажчиків). Наприклад:
struct Node { Data d; //тип Data повинен бути визначений раніше
Node *p; }
Node *newPtr;
Основні операції в С++ new , delete :
newPtr = new Node;
delete newPtr;
Звернення до даних:
newPtr -> d або (*newPtr).d

Слайд 6

Інструментарій С++

Для роботи з адресами пам`яті С++ пропонує досить чисельний набір інструментів.
Розглядали

Інструментарій С++ Для роботи з адресами пам`яті С++ пропонує досить чисельний набір
раніше операції (повторити):
унарні: * -(розіменування), & -(отримання адреси), new , delete -(виділення та звільнення пам`яті);
бінарні: ., -> (отримання поля, розіменування поля).
Стандартна бібліотека () надає функції для керування розподілом пам`яті (самостійно):
calloc(), malloc(), realloc() – виділення;
free() – звільнення.
В С++ рекомендовано для керування розподілом пам`яті використовувати операції new та delete .

Слайд 7

Лінійні зв`язані списки

Найпростіші динамічні структури даних – лінійні зв`язані списки. Як

Лінійні зв`язані списки Найпростіші динамічні структури даних – лінійні зв`язані списки. Як
мінімум зберігається зв`язок поточного вузла з наступним. Список задається покажчиком на початковий вузол.

Інф.1

Інф.2

Інф.n Ø

Покажчик

!!! Втрата покажчиків (розрив зв`язків) призведе до недоступності відповідних елементів, або списку в цілому (поява “сміття”).

Слайд 8

Приклад

Потрібно:
побудувати зв`язний список з трьох елементів <1, 2, 3>;
вивести інформацію, що

Приклад Потрібно: побудувати зв`язний список з трьох елементів ; вивести інформацію, що
розташована в елементах списку;
звільнити пам`ять, прибравши елементи списку.

p

1

2

3

Ø

В прикладі розглянемо покрокове виконання всіх вказаних дій зі списком.

Pr_1

Слайд 9

Представлення даних

typedef struct Node {int dat;
Node *next;} Listn, *Listp;
!!! Не єдиний

Представлення даних typedef struct Node {int dat; Node *next;} Listn, *Listp; !!!
можливий спосіб визначити потрібні типи даних.

Слайд 10

Побудова списку

//побудова списку з 3 елементів
Listp lform(int d1, int d2, int d3){

Побудова списку //побудова списку з 3 елементів Listp lform(int d1, int d2,
Listp p, t;
p = new Listn; p->dat = d3; p->next = NULL;
t = new Listn; t->dat = d2; t->next = p; p = t;
t = new Listn; t->dat = d1; t->next = p; p = t;
return p;
}

Слайд 11

Відображення списку

//відображення 3-елементного списку
void lprint(Listp p){
cout << p->dat << ' ';

Відображення списку //відображення 3-елементного списку void lprint(Listp p){ cout dat next; cout
p = p->next;
cout << p->dat << ' '; p = p->next;
cout << p->dat << endl;
}

Слайд 12

Вилучення списку

//знищення 3-елементного списку
void ldel(Listp p){
Listp t;
t = p->next; delete

Вилучення списку //знищення 3-елементного списку void ldel(Listp p){ Listp t; t =
p; p = t;
t = p->next; delete p; p = t;
delete p;
}

Слайд 13

Альтернативи

Для представлення списку використовувати масив.

1

2

3

Порівнянні наведених способів.
???

Не є проблемою записати відповідний

Альтернативи Для представлення списку використовувати масив. 1 2 3 Порівнянні наведених способів.
аналог розглянутим діям.

Слайд 14

Альтернативи

Для представлення списку використовувати масив.

1

2

3

Порівнянні наведених способів:

фіксований розмір масиву → обмеження

Альтернативи Для представлення списку використовувати масив. 1 2 3 Порівнянні наведених способів:
на розмір списку, ефективність використання наявної пам`яті;
зв`язки, порядок елементів списку визначається розташуванням у масиві → зміни – передбачають переміщення елементів;
масив займає неперервну область пам`яті;
не завжди мови програмування надають явні можливості роботи з адресами.

Слайд 15

Альтернативи

m

n

k

2

1

3

k

m

Ø

n

beg

Порівнянні наведених способів.
???

Pr_2

Використовує явно лише можливості роботи з масивами та їх

Альтернативи m n k 2 1 3 k m Ø n beg
елементами.

Слайд 16

Приклад

Використовують також списки з двома зв`язками.

Ø

Ø

pbeg

pend

Потрібно:
побудувати подібний список з N елементів <1,

Приклад Використовують також списки з двома зв`язками. Ø Ø pbeg pend Потрібно:
2, …, N>;
додати новий елемент з заданим значенням після елемента з вказаним значенням (пошук місця вставки);
вилучити елемент з заданим значенням (пошук ел.);
вивести список на екран;
знищити список, звільнивши пам`ять.

Pr_3

Їх переваги, вади ???

Слайд 17

Представлення даних

struct Node {int dat;
Node *next;
Node *prev;};
!!! Не єдиний можливий

Представлення даних struct Node {int dat; Node *next; Node *prev;}; !!! Не
спосіб визначити потрібні типи даних.

Слайд 18

Побудова списку

//формування першого елемента списку
Node *first(int d){
Node *pv = new Node;
pv->dat =

Побудова списку //формування першого елемента списку Node *first(int d){ Node *pv =
d; pv->next = NULL; pv->prev = NULL;
return pv;
}
//додавання елементів в кінець списку 2, 3, ..., nn
void add(Node **pend, int d){
Node *pv = new Node;
pv->dat = d; pv->next = NULL; pv->prev = *pend;
(*pend)->next = pv;
*pend = pv;
}

Слайд 19

Пошук у списку

//пошук елемента за ключем
Node *find(Node *const pbeg, int key){
Node

Пошук у списку //пошук елемента за ключем Node *find(Node *const pbeg, int
*pv = pbeg;
while (pv){
if (pv->dat == key) break;
pv = pv->next;
}
return pv;
}

Слайд 20

Вставка елемента у список

Node *insert(Node *const pbeg, Node **pend, int key, int

Вставка елемента у список Node *insert(Node *const pbeg, Node **pend, int key,
d)
{if (Node *pkey = find(pbeg, key)) {
Node *pv = new Node;
pv->dat = d;
pv->next = pkey->next; //зв`язок нового вузла з наступним
pv->prev = pkey; //зв`язок нового вузла з попереднім
pkey->next = pv; //зв`язок попереднього з новим вузлом
//зв`язок наступного з новим вузлом
if (pkey != *pend) (pv->next)->prev = pv;
else *pend = pv; //якщо вузол стає останнім
return pv;}
return NULL; //місце для вставки не було знайдено
}

Слайд 21

Вставка елемента у список

Вставка елемента у список

Слайд 22

Вилучення елемента зі списку

//вилучення елемента
bool remove(Node **pbeg, Node **pend, int key){
if

Вилучення елемента зі списку //вилучення елемента bool remove(Node **pbeg, Node **pend, int
(Node *pkey = find(*pbeg, key)){
if (pkey == *pbeg) { *pbeg = (*pbeg)->next;
(*pbeg)->prev = NULL;}
else if (pkey == *pend) { *pend = (*pend)->prev;
(*pend)->next = NULL;}
else { (pkey->prev)->next = pkey->next;
(pkey->next)->prev = pkey->prev;}
delete pkey;
return true;}
return false;
}

Слайд 23

Вилучення елемента зі списку

pkey

pkey->prev

pkey->next

Вилучення елемента зі списку pkey pkey->prev pkey->next

Слайд 24

Відображення списку

//виведення списку
void lprint(Node *pbeg){
Node *pv = pbeg;
while (pv){
cout << pv->dat <<

Відображення списку //виведення списку void lprint(Node *pbeg){ Node *pv = pbeg; while
' ';
pv = pv->next;
} cout << endl;
}

Слайд 25

Вилучення списку

//знищення списку
void ldel(Node *pbeg){
Node *pv;
while (pbeg){
pv = pbeg;
pbeg = pbeg->next;
delete pv;
}
}

Вилучення списку //знищення списку void ldel(Node *pbeg){ Node *pv; while (pbeg){ pv

Слайд 26

Зауваження

Стандартна бібліотека С++ надає також інші можливості керування розподілом пам`яті.
Не відбувається автоматичне

Зауваження Стандартна бібліотека С++ надає також інші можливості керування розподілом пам`яті. Не
повернення виділеної пам`яті, що може стати причиною появи “сміття”, “переповнення” пам`яті.
Операція delete звільняє пам`ять (яка може бути розподілена у подальшому), але не прибирає й не змінює сам покажчик.
Динамічні зв`язані структури дозволяють використовувати для представлення великих даних не обов`язково неперервну область пам`яті.

Слайд 27

Підсумки

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

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

Слайд 28

Поради

Одразу звільняти пам`ять (операція - delete), що була виділена (операцією new) та

Поради Одразу звільняти пам`ять (операція - delete), що була виділена (операцією new)
стала непотрібною.
Писати “прозорі”, структуровані програми.
Здійснювати внутрішнє документування у програмі за допомогою коментарів.
Не зловживати “трюкачеством”.
Розумним чином “форматувати” текст програми.
Принаймні на початковому етапі не нехтувати можливостями візуально відслідковувати дії, що відбуваються з динамічною структурою.
При тестуванні та налагодженні додатково відображати проміжні стани динамічних структур.

Слайд 29

Задачі

Лінійний зв`язний список містить послідовність цілих чисел. Написати функцію для:
знаходження максимального з

Задачі Лінійний зв`язний список містить послідовність цілих чисел. Написати функцію для: знаходження
чисел;
знаходження кількості чисел у послідовності;
перевірки належності заданого числа до послідовності;
перевірки наявності двох однакових чисел у послідовності.

Слайд 30

Задачі

Рядки тексту являють собою прізвища, які можуть повторюватися. Потрібно прочитати текст і

Задачі Рядки тексту являють собою прізвища, які можуть повторюватися. Потрібно прочитати текст
надрукувати кожне прізвище по одному разу. Порядок прізвищ не має значення. Проблеми:
джерело інформації - файл (текстовий);
кількість прізвищ не відома.
Порада – для тимчасового внутрішнього збереження інформації скористатися динамічною структурою.
Имя файла: Динамічні-структури-даних.pptx
Количество просмотров: 169
Количество скачиваний: 1