Podstawy programowania w języku C++

Podobne dokumenty
Podstawy programowania w języku C++

Programowanie w języku C++

Języki programowania obiektowego Nieobiektowe elementy języka C++

Wprowadzenie do programowania w języku C

DYNAMICZNE PRZYDZIELANIE PAMIECI

Co to jest sterta? Sterta (ang. heap) to obszar pamięci udostępniany przez system operacyjny wszystkim działającym programom (procesom).

Podstawy programowania w języku C++

Lab 9 Podstawy Programowania

Podstawy programowania komputerów

Wskaźniki. Przemysław Gawroński D-10, p marca Wykład 2. (Wykład 2) Wskaźniki 8 marca / 17

Wskaźniki. Informatyka

Podstawy programowania. Wykład 6 Wskaźniki. Krzysztof Banaś Podstawy programowania 1

Temat: Dynamiczne przydzielanie i zwalnianie pamięci. Struktura listy operacje wstawiania, wyszukiwania oraz usuwania danych.

Wskaźniki i dynamiczna alokacja pamięci. Spotkanie 4. Wskaźniki. Dynamiczna alokacja pamięci. Przykłady

Podstawy programowania w języku C++

Wskaźniki. nie są konieczne, ale dają językowi siłę i elastyczność są języki w których nie używa się wskaźników typ wskaźnikowy typ pochodny:

Podstawy programowania w języku C i C++

KURS C/C++ WYKŁAD 6. Wskaźniki

Wykład 1: Wskaźniki i zmienne dynamiczne

ZASADY PROGRAMOWANIA KOMPUTERÓW

Struktury czyli rekordy w C/C++

Podstawy programowania w języku C++

Podstawy programowania w języku C++

// Liczy srednie w wierszach i kolumnach tablicy "dwuwymiarowej" // Elementy tablicy są generowane losowo #include <stdio.h> #include <stdlib.

Wskaźniki w C. Anna Gogolińska

Podstawy programowania w języku C++

Programowanie w C++ Wykład 5. Katarzyna Grzelak. 16 kwietnia K.Grzelak (Wykład 1) Programowanie w C++ 1 / 27

Język C++ Różnice między C a C++

Programowanie w języku C++

Podstawy programowania w języku C++

> C++ dynamiczna alokacja/rezerwacja/przydział pamięci. Dane: Iwona Polak. Uniwersytet Śląski Instytut Informatyki

METODY I JĘZYKI PROGRAMOWANIA PROGRAMOWANIE STRUKTURALNE. Wykład 02

Podstawy programowania skrót z wykładów:

Języki programowania. Przetwarzanie plików amorficznych Konwencja języka C. Część siódma. Autorzy Tomasz Xięski Roman Simiński

Wykład 4: Klasy i Metody

Podstawy programowania

Spis treści WSKAŹNIKI. DYNAMICZNY PRZYDZIAŁ PAMIĘCI W JĘZYKU C. Informatyka 2. Instrukcja do pracowni specjalistycznej z przedmiotu

Zaawansowane programowanie w języku C++ Zarządzanie pamięcią w C++

Techniki Programowania wskaźniki

Wprowadzenie do programowania w języku C

Laboratorium nr 9. Temat: Wskaźniki, referencje, dynamiczny przydział pamięci, tablice dynamiczne. Zakres laboratorium:

Języki i metodyka programowania. Wskaźniki i tablice.

Wskaźnik może wskazywać na jakąś zmienną, strukturę, tablicę a nawet funkcję. Oto podstawowe operatory niezbędne do operowania wskaźnikami:

1. Wartość, jaką odczytuje się z obszaru przydzielonego obiektowi to: a) I - wartość b) definicja obiektu c) typ oboektu d) p - wartość

Język C zajęcia nr 11. Funkcje

Języki programowania obiektowego Nieobiektowe elementy języka C++

Wprowadzenie do programowanie obiektowego w języku C++

Wstęp do programowania

Wstęp do wskaźników w języku ANSI C

Informacje ogólne. Karol Trybulec p-programowanie.pl 1. 2 // cialo klasy. class osoba { string imie; string nazwisko; int wiek; int wzrost;

JĘZYKI PROGRAMOWANIA Z PROGRAMOWANIEM OBIEKTOWYM. Wykład 6

Spis treści WSKAŹNIKI. DYNAMICZNY PRZYDZIAŁ PAMIĘCI W JĘZYKU C. Informatyka 2. Instrukcja do pracowni specjalistycznej z przedmiotu

Języki programowania. Przetwarzanie tablic znaków. Część druga. Autorzy Tomasz Xięski Roman Simiński

Podstawy programowania w języku C++

W2 Wprowadzenie do klas C++ Klasa najważniejsze pojęcie C++. To jest mechanizm do tworzenia obiektów. Deklaracje klasy :

Tablice, funkcje - wprowadzenie

TEMAT : KLASY DZIEDZICZENIE

Wstęp do programowania

Globalne / Lokalne. Wykład 15. Podstawy programowania (język C) Zmienne globalne / lokalne (1) Zmienne globalne / lokalne (2)

Szablony klas, zastosowanie szablonów w programach

Podstawy informatyki. Elektrotechnika I rok. Język C++ Operacje na danych - wskaźniki Instrukcja do ćwiczenia

Tbli Tablice obiektów biktó są tworzone dokładnie d tak samo, jak i tablice, składające się z elementów innego typu

Wykład II. Programowanie II - semestr II Kierunek Informatyka. dr inż. Janusz Słupik. Wydział Matematyki Stosowanej Politechniki Śląskiej

Informatyka I. Klasy i obiekty. Podstawy programowania obiektowego. dr inż. Andrzej Czerepicki. Politechnika Warszawska Wydział Transportu 2018

Dariusz Brzeziński. Politechnika Poznańska, Instytut Informatyki

Podstawy programowania. Wykład Funkcje. Krzysztof Banaś Podstawy programowania 1

Spis treści JĘZYK C - WSKAŹNIKI, DYNAMICZNY PRZYDZIAŁ PAMIĘCI. Informatyka 2. Instrukcja do pracowni specjalistycznej z przedmiotu

Zmienne i struktury dynamiczne

Podstawy programowania w języku C++

1 Wskaźniki i zmienne dynamiczne, instrukcja przed zajęciami

Programowanie w C++ Wykład 5. Katarzyna Grzelak. 26 marca kwietnia K.Grzelak (Wykład 1) Programowanie w C++ 1 / 40

Wskaźniki. Programowanie Proceduralne 1

Programowanie w C++ Wykład 4. Katarzyna Grzelak. 19 marca K.Grzelak (Wykład 1) Programowanie w C++ 1 / 37

Zmienne, stałe i operatory

Języki programowania obiektowego Nieobiektowe elementy języka C++

Wykład 8: klasy cz. 4

PROE wykład 2 operacje na wskaźnikach. dr inż. Jacek Naruniec

Wstęp do programowania INP001213Wcl rok akademicki 2018/19 semestr zimowy. Wykład 4. Karol Tarnowski A-1 p.

Referencje do zmiennych i obiektów

wykład III uzupełnienie notatek: dr Jerzy Białkowski Programowanie C/C++ Język C - zarządzanie pamięcią, struktury,

IX. Wskaźniki.(3 godz.)

Programowanie obiektowe W3

Adam Kotynia, Łukasz Kowalczyk

Programowanie i struktury danych

Konwersje napis <-> liczba Struktury, unie Scanf / printf Wskaźniki

Tablice, funkcje, wskaźniki - wprowadzenie

Materiał Typy zmiennych Instrukcje warunkowe Pętle Tablice statyczne Wskaźniki Tablice dynamiczne Referencje Funkcje

Stałe, tablice dynamiczne i wielowymiarowe

C++ - klasy. C++ - klasy. C++ - klasy. C++ - klasy. C++ - klasy KONSTRUKTORY

Uzupełnienie dot. przekazywania argumentów

Programowanie obiektowe Wykład 3. Dariusz Wardowski. dr Dariusz Wardowski, Katedra Analizy Nieliniowej, WMiI UŁ 1/21

Podstawy programowania. Wykład: 12. Struktury, unie, pola bitowe. dr Artur Bartoszewski -Podstawy programowania, sem 1 - WYKŁAD

Dr inż. Grażyna KRUPIŃSKA. D-10 pokój 227 WYKŁAD 7 WSTĘP DO INFORMATYKI

referencje Wykład 2. Programowanie (język C++) Referencje (1) int Num = 50; zdefiniowano zmienną Num (typu int) nadając jej wartość początkową 50.

Podstawy algorytmiki i programowania - wykład 4 C-struktury

Podstawy programowania. Wykład 6 Złożone typy danych: struktury, unie. Krzysztof Banaś Podstawy programowania 1

C++ - klasy. C++ - klasy. C++ - klasy. C++ - klasy. C++ - klasy PRAWA PRZYJACIÓŁ KLASY. Dostęp z zewnątrz: Dostęp z wewnątrz:

Dynamiczny przydział pamięci w języku C. Dynamiczne struktury danych. dr inż. Jarosław Forenc. Metoda 1 (wektor N M-elementowy)

Strona główna. Strona tytułowa. Programowanie. Spis treści. Sobera Jolanta Strona 1 z 26. Powrót. Full Screen. Zamknij.

IMIĘ i NAZWISKO: Pytania i (przykładowe) Odpowiedzi

Transkrypt:

Podstawy programowania w języku C++ Część ósma Zmienne wskaźnikowe koncepcja, podstawowe zastosowania Autor Roman Simiński Kontakt roman.siminski@us.edu.pl www.us.edu.pl/~siminski Niniejsze opracowanie zawiera skrót treści wykładu, lektura tych materiałów nie zastąpi uważnego w nim uczestnictwa. Opracowanie to jest chronione prawem autorskim. Wykorzystywanie jakiegokolwiek fragmentu w celach innych niż nauka własna jest nielegalne. Dystrybuowanie tego opracowania lub jakiejkolwiek jego części oraz wykorzystywanie zarobkowe bez zgody autora jest zabronione.

Co to jest zmienna przypomnienie Zmienna jest obiektem w programie, rezydującym w pamięci operacyjnej, przeznaczonym do przechowywania wartości pewnego typu. Każda zmienna ma swoją nazwę, oraz typ wartości. Zmienne są przechowywane w pamięci operacyjnej, liczba zajętych bajtów zależy od typu zmiennej. Nazwa zmiennej identyfikuje zmienną w programie zwalniając programistę od zastanawiania się, pod jakim adresem w pamięci zmienna jest zlokalizowana. Adres w pamięci... Nazwa zmiennej int i; i = 10; 02c4a45fh 10 i Wartość zmiennej... Copyright Roman Simiński Strona : 2

Dziwne pojęcia l-wartość i r-wartość Obiekt jest pewnym nazwanym obszarem pamięci. Pod pojęciem l-wartości rozumiemy wyrażenie lokalizujące ten obiekt w pamięci Zmienna może występować po lewej stronie operatora przypisania, mówi się, że jest wtedy l-wartością. Wszystko, co może występować po prawej stronie operatora przypisania jest r-wartością. int i; int j; j = 5; i = j; 5 = i; j to l-wartość 5 to r-wartość i to l-wartość j to r-wartość Nie każda r-wartość to l-wartość Copyright Roman Simiński Strona : 3

Zmienne wskaźnikowe motywacja do nauki W języku C intensywnie wykorzystuje się l-wartości oparte na zmiennych wskaźnikowych oraz na wyrażeniach te zmienne zawierających. Dokładne opanowanie zasad posługiwania się wskaźnikami jest niezbędne do efektywnego i sprawnego programowania w C i C++. Tej umiejętności nie można pominąć, przeskoczyć lub zostawić na później. Nie oszukujmy się ten, kto nie opanuje zasad posługiwania się wskaźnikami nigdy nie będzie prawdziwym, profesjonalnym programistą, wykorzystującym język C lub C++. Koncepcja wskaźników oraz metody ich wykorzystania są proste. Wymagają one jednak uwagi, zrozumienia i myślenia. Copyright Roman Simiński Strona : 4

Po co są zmienne wskaźnikowe? Zmienna wskaźnikowa przeznaczona jest do lokalizowania (inaczej wskazywania) obiektów w pamięci operacyjnej. Jedyną rolą zmiennej wskaźnikowej jest umożliwienie odwoływania się do obiektów wskazywanych. Pamięć operacyjna Zmienna wskaźnikowa Obiekt wskazywany Copyright Roman Simiński Strona : 5

Po co są zmienne wskaźnikowe? Zmienna wskaźnikowa rezyduje w pamięci operacyjnej. Zmienna wskaźnikowa może lokalizować w pamięci operacyjnej inne zmienne, nienazwane bloki pamięci oraz bloki zawierające kod programu, np. funkcje. Sama zmienna wskaźnikowa może być również wskazywana przez inną zmienną wskaźnikową. Zmienna wskaźnikowa Pamięć operacyjna Obiekt wskazywany Zmienna wskaźnikowa Copyright Roman Simiński Strona : 6

Trzy stany zmiennej wskaźnikowej Zmienna wskaźnikowa wskazuje na konkretny obiekt w pamięci: Pamięć operacyjna Zmienna wskaźnikowa Obiekt wskazywany OK Zmienna wskaźnikowa nie wskazuje na żaden obiekt: Pamięć operacyjna Zmienna wskaźnikowa OK Zmienna wskaźnikowa wskazuje na nie wiadomo co: Pamięć operacyjna Zmienna wskaźnikowa? Kiepsko Copyright Roman Simiński Strona : 7

Co zawiera zmienna wskaźnikowa? Zwykle przyjmuje się, że zmienna wskaźnikowa zawiera w sobie adres obiektu wskazywanego. Jednak zmienna wskaźnikowa nie musi w sobie zawierać adresu bezpośredniego (fizycznego). Zawartość zmiennej wskaźnikowej może zawierać inną informację, pozwalającą na precyzyjne i jednoznaczne zidentyfikowanie położenia obiektu w pamięci. Pamięć operacyjna Zmienna wskaźnikowa 345fa012h Obiekt wskazywany Adres: 345fa012h Copyright Roman Simiński Strona : 8

Przykład implementacji zmiennej wskaźnikowej Intel 8086 Zmienna wskaźnikowa zawiera przesunięcie (ang. offset) obiektu względem początku segmentu gdy wskaźniki są krótkie (ang. near) odwołania wewnątrz segmentu. Przesunięcie Adres segmentu Segment 02c4a45fh +0 +1 +2 +3 +4 Obiekt Krótka zmienna wskaźnikowa 0003 Copyright Roman Simiński Strona : 9

Przykład implementacji zmiennej wskaźnikowej Intel 8086 Zmienna wskaźnikowa zawiera adres segmentu i przesunięcie obiektu gdy wskaźniki są długie (ang. far) odwołania międzysegmentowe. Przesunięcie Adres segmentu Segment 02c4a45fh +0 +1 +2 +3 +4 Obiekt Długa zmienna wskaźnikowa 02c4a45f:0003 Copyright Roman Simiński Strona : 10

Deklaracja zmiennej wskaźnikowej Deklarowana zmienna będzie wskaźnikiem, kompilator wie, ile dla niej zarezerwować pamięci. To oznacza, że deklarowana zmienna wskaźnikowa będzie przeznaczona do lokalizowania w pamięci obiektów typu int. int * pi ; Nazwa deklarowanej zmiennej wskaźnikowej zbudowana wg. zwykłych reguł, często zawiera p lub ptr od pointer. int i = 10; int * pi; int i = 10; int * pi = NULL; Nazwa zmiennej Nazwa zmiennej i 10 i 10 Wartość zmiennej Wartość zmiennej pi?? pi Copyright Roman Simiński Strona : 11

Rola wskaźnika pustego NULL Tak zdefiniowana zmienna wskaźnikowa: int * pi; pi?? ma wartość początkową zależną od kontekstu deklaracji. Jeżeli ta zmienna jest klasy auto, to jej wartość jest przypadkowa zmienna wskazuje zatem na bliżej nieznany obiekt w pamięci. W pliku nagłówkowym stddef.h zdefiniowana stałą NULL, reprezentującą wskaźnik pusty, niezależny od platformy i implementacji. Tak zdefiniowana zmienna: int * pi = NULL; pi jest wskaźnikiem pustym, a więc nie wskazuje żadnego obiektu w pamięci. Copyright Roman Simiński Strona : 12

Wartość NULL kontra 0 Stała NULL jest definiowana jako wartość 0 lub 0L. Można zatem zamiast wartością NULL, posługiwać się wartością 0. W języku C praktykuje się stosowanie wartości NULL a nie wartości 0. W języku C++ praktykuje się stosowanie wartości 0 zamiast NULL. Niezależnie od przyjętej wartości wskaźnika pustego, jawnie inicjowanie zmiennych wskaźnikowych oraz posługiwanie się wartością pustą dla wskaźników niezakotwiczonych jest dobrą praktyką programistyczną w języku C i C++. To, czy zmienna wskaźnikowa jest wskaźnikiem pustym można sprawdzić: if( pi!= NULL ) // Tu jakie ś operacje na obiekcie // wskazywanym przez pi if( pi == NULL ) // Nie odwołujemy si ę do obiektu // wskazywanego przez pi nie ma go! Copyright Roman Simiński Strona : 13

Przypisywanie wartości zmiennym wskaźnikowym int i = 10; int * pi = NULL; i 10 int i = 10; int * pi = NULL; pi = &i; i 10 pi pi Od momentu tego przypisania, pi wskazuje zmienną i, umożliwiając realizację dowolnych operacji na tej zmiennej. pi = & i ; Wyrażenie wskaźnikowe lokalizujące zmienną i w pamięci. Jednoargumentowy operator & buduje wyrażenie wskaźnikowe lokalizujące zmienną w pamięci operacyjnej. Argument musi być l-wartością, nie odnoszącą się do obiektu register ani pola bitowego. Copyright Roman Simiński Strona : 14

Odwoływanie sie do obiektu wskazywanego int i = 10; int * pi = NULL; pi = &i; *pi = 20; i 20 *pi == i pi * pi = 20 ; Ten zapis oznacza obiekt wskazywany przez pi. Zapis *pi może wystąpić wszędzie tam, gdzie może wystąpić i. Jednoargumentowy operator adresowania pośredniego *, daje w wyniku obiekt wskazywany przez argument pi. Dowolne wyrażenie typu zgodnego z typem obiektu wskazywanego. Copyright Roman Simiński Strona : 15

Odwoływanie sie do obiektu wskazywanego, uwagi Po przypisaniu: pi = &i; te fragmenty kodu są równoważne: cin >> *pi; if( *pi == 0 ) cout << "Bledna wartosc"; else y = *pi * x; cout << "Wynik: " << y; cin >> i; if( i == 0 ) cout << "Bledna wartosc"; else y = i * x; cout << "Wynik: " << y; Jeżeli wskaźnik pi wskazuje na zmienną i, to *pi może wystąpić wszędzie tam, gdzie może wystąpić i. Zmienna pi jest linkiem (odnośnikiem) do zmiennej i, a wyrażenie *pi jest aliasem (alternatywną nazwą) zmiennej i. Copyright Roman Simiński Strona : 16

Typowe zastosowania zmiennych wskaźnikowych Realizacja przekazywania parametrów przez zmienną. Wykorzystanie pamięci zarządzanej dynamicznie. Manipulowanie tablicami. Budowa rekurencyjnych struktur danych. Copyright Roman Simiński Strona : 17

Przypomnienie: przekazywanie parametrów przez wartość void inc( int i ) i = i + 1; Jaka jest wartość a po wywołaniu inc? int a = 5; inc( a ); cout << a; Przed wywołaniem inc( a ) Wywołanie inc( a ) Wykonanie inc( a ) Po wykonaniu inc( a ) a 5 a 5 a 5 a 5 i 5 i 6 5X i=i+1 Copyright Roman Simiński Strona : 18

Wskaźniki a przekazywanie parametrów void inc( int * i ) *i = *i + 1; Jaka jest wartość a po wywołaniu inc? int a = 5; inc( &a ); cout << a; Parametr formalny i jest wskaźnikiem. Parametr aktualny wywołania również. Przed wywołaniem inc( a ) Wywołanie inc( a ) a 5 a 5 Wykonanie *i == a inc( a ) *i == a a 6 5X Po wykonaniu inc( a ) a 56 i i *i=*i+1 Copyright Roman Simiński Strona : 19

Wskaźniki typu void * Typ void * oznacza wskazanie niezwiązane z żadnym typem. Wskaźnik takiego typu może wskazywać daną dowolnego typu. float f = 2.5; int i = 5; char c = 'A'; void * ptr; ptr = &f; ptr = &i; Wskaźnik ptr może pokazywać na obiekty różnych typów. ptr = &c; Copyright Roman Simiński Strona : 20

Wskaźniki typu void *, cd.... Aby odwołać się do obiektu wskazywanego, należy poinformować kompilator jaki jest jego typ, dokonując konwersji (tzw. rzutowania) typu wskaźnika. void * ptr; ptr = &f; cout << endl << *( ( float * ) ptr ); Rzutowanie wskaźnika ptr wskazanie na obiekt typu float. void * ptr; ptr = &f; cout << endl << * ( float * ) ptr; Copyright Roman Simiński Strona : 21

Wskaźniki typu void *, cd.... Wskaźnik void * można rzutować na różne typy: float f = 2.5; int i = 5; char c = 'A'; void * ptr; ptr = &f; cout << endl << *( ( float * )ptr ); ptr = &i; cout << endl << *( ( int * )ptr ); ptr = &c; cout << endl << *( ( char * )ptr ); Copyright Roman Simiński Strona : 22

Pojęcie sterty, sterta a stos Sterta (ang. heap) to wydzielony obszar pamięci wolnej: przeznaczony do przechowywania danych dynamicznych, kontrolowany ręcznie przez programistę, ograniczony pod względem rozmiaru, Sterta przydzielany pasującymi fragmentami. Stos (ang. stack) to wydzielony obszar pamięci roboczej: Dane przeznaczony do przechowywania danych automatycznych, nie jest bezpośrednio kontrolowany przez programistę, Stos ograniczony pod względem rozmiaru, przydzielany wg. zasady LIFO (ang. last in, first out). Copyright Roman Simiński Strona : 23

Dynamiczny przydział pamięci Dynamiczny przydział pamięci polega na zarezerwowaniu fragmentu pamięci w obszarze pamięci wolnej (sterty), dla obiektu pamięciowego zwanego dynamicznym. Typowy scenariusz wykorzystania dynamicznego przydziału pamięci: Określenie wielkości potrzebnego obszaru pamięci. Przydział pamięci i zapamiętanie wskazania tego obszaru w zmiennej wskaźnikowej. Sprawdzenie czy przydział pamięci się powiódł, jeżeli tak to: Wykorzystanie przydzielonego bloku pamięci. Zwolnienie przydzielonego bloku pamięci, gdy nie jest już potrzebny. Copyright Roman Simiński Strona : 24

Dynamiczny przydział pamięci w języku C Przydział pamięci realizują funkcje: void * malloc( size_t size ) void * calloc( size_t nitems, size_t size ) void * realloc( void * ptr, size_t size ) Obszary pamięci przydzielone tymi funkcjami należało zwolnić funkcją: void free( void * ptr ) Funkcje zarządzające przydziałem/zwalnianiem bloków pamięci operują na wskaźnikach void *. Przydzielane bloki są amorficzne są to kawałki pamięci o rozmiarze liczonym w bajtach. Wykorzystanie powyższych funkcji wymaga włączenia pliku nagłówkowego dyrektywą #include <stdlib.h> lub #include <cstdlib> w C++. Copyright Roman Simiński Strona : 25

Dynamiczny przydział pamięci w języku C etap I Definicja zmiennej wskaźnikowej p, zainicjowanej wskaźnikiem pustym. int main() int * p = NULL; Sterta p = malloc( sizeof( int ) ); if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); p Dane Stos Copyright Roman Simiński Strona : 26

Dynamiczny przydział pamięci w języku C etap II Funkcja malloc przydziela na stercie blok pamięci o rozmiarze sizeof(int). Rezultatem funkcji jest wskaźnik do przydzielonego obszaru lub NULL, jeżeli pamięć nie może być przydzielona. int main() int * p = NULL; Sterta p = malloc( sizeof( int ) ); if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); p Dane Stos Copyright Roman Simiński Strona : 27

Dynamiczny przydział pamięci w języku C etap III Zawsze należy sprawdzić poprawność przydziału pamięci. Odwołanie do wskaźnika pustego jest błędem! int main() int * p = NULL; Sterta p = malloc( sizeof( int ) ); if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); p Dane Stos Copyright Roman Simiński Strona : 28

Dynamiczny przydział pamięci w języku C etap IV Wykorzystanie przydzielonego bloku pamięci. Ponieważ zmienna wskaźnikowa p jest skojarzona z typem int, przydzielony obszar traktowany jest jak dana typu int. int main() int * p = NULL; p = malloc( sizeof( int ) ); 10 *p Sterta if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); p Dane Stos Copyright Roman Simiński Strona : 29

Dynamiczny przydział pamięci w języku C etap IV, cd.... Z obiektem wskazywanym przez zmienną p można robić wszystko to, co dozwolone dla danej typu int. Wyrażenie ++(*p) zwiększa obiekt wskazywany przez zmienną p. int main() int * p = NULL; p = malloc( sizeof( int ) ); 11 *p Sterta if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); p Dane Stos Copyright Roman Simiński Strona : 30

Dynamiczny przydział pamięci w języku C etap V Wywołanie funkcji free powoduje zwolnienie bloku pamięci wskazywanego przez p, blok ten zwracany jest do puli bloków wolnych. Uwaga po wywołaniu free wskaźnik p dalej pokazuje na zwolniony blok pamięci! int main() int * p = NULL; Sterta p = malloc( sizeof( int ) ); if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); p Dane Stos Copyright Roman Simiński Strona : 31

Dynamiczny przydział pamięci w języku C uwagi Mimo, że po wywołaniu free wskaźnik p dalej pokazuje na zwolniony obszar, próba odwołania się do niego jest błędem. Ten obszar być może został właśnie przydzielony ponownie. int main() int * p = NULL; p = malloc( sizeof( int ) );??? Sterta if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); *p = 0; *p = 100; Błąd. Odwołanie do zwolnionego lub nieprzydzielonego bloku p Dane Stos Copyright Roman Simiński Strona : 32

Dynamiczny przydział pamięci w języku C uwagi, cd.... Dobrą praktyką jest zerowanie zmiennych wskaźnikowych na etapie ich deklaracji, po zwolnieniu pamięci oraz testowanie czy wskaźnik nie jest pusty przed odwołaniem do obiektu wskazywanego. int main() int * p = NULL; Sterta p = malloc( sizeof( int ) ); if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); p = NULL; p Dane Stos Copyright Roman Simiński Strona : 33

Dynamiczny przydział pamięci w języku C opis funkcji void * malloc( size_t size ); Rezultatem funkcji malloc jest wskaźnik do obszaru pamięci przeznaczonego dla obiektu o rozmiarze size. Rezultatem jest NULL, jeżeli polecenie nie może być zrealizowane. Obszar nie jest inicjowany. void * calloc( size_t nitems, size_t size ); Rezultatem funkcji calloc jest wskaźnik do obszaru pamięci przeznaczonego dla nitems obiektów o rozmiarze size. Rezultatem jest NULL, jeżeli polecenie nie może być zrealizowane. Obszar jest inicjowany zerami. Copyright Roman Simiński Strona : 34

Dynamiczny przydział pamięci w języku C opis funkcji void * realloc( void * ptr, size_t size ); Funkcja dokonuje próby zmiany rozmiaru bloku wskazywanego przez ptr, który był poprzednio przydzielony wywołaniem funkcji calloc lub malloc. Zawartość wskazywanego obszaru pozostaje niezmieniona. Jeżeli nowy rozmiar jest większy od poprzednio przydzielonego, dodatkowe bajty mają nieokreśloną wartość. Jeżeli nowy rozmiar jest mniejszy, bajty z różnicowego obszaru są zwalniane. Jeżeli ptr == NULL to funkcja działa jak malloc. Rezultatem funkcji jest wskaźnik na obszar pamięci o nowym rozmiarze (może być ulokowany w pamięci w innej lokalizacji niż poprzednio). Rezultatem jest NULL w przypadku błędu lub próby przydziału bloku o zerowym rozmiarze. void free( void * ptr ); Zwalnia obszar pamięci wskazywany przez ptr. Parametr musi być wskaźnikiem do obszaru pamięci przydzielonego uprzednio przez malloc, calloc lub realloc. Copyright Roman Simiński Strona : 35

Dynamiczny przydział pamięci w języku C dla int nie ma sensu... Dynamiczny przydział pamięci dla pojedynczych danych typu int, char czy double najczęściej nie ma sensu. Ale ma sens dla obiektów zajmujących dużo pamięci operacyjnej oraz dla złożonych struktur danych. int main() bigtype * p = NULL; p = malloc( sizeof( bigtype ) ); Dana typu bigtype Sterta if( p!= NULL ) operacje na dużym obiekcie free( p ); p = NULL; p Dane Stos Copyright Roman Simiński Strona : 36

Dynamiczny przydział pamięci w języku C++ W języku C++ można używać funkcji malloc, calloc,.... Jednak w tym języku wprowadzono specjalne operatory zarządzające pamięcią: new oraz delete. Ich wykorzystanie jest zalecane a w wielu przypadkach konieczne. Operatory są częścią języka i współpracują z mechanizmami kontroli typów. Zatem w języku C++ zarządzanie pamięcią dynamiczną realizować będzie: operator new przydział pamięci, operator delete zwolnienie pamięci. Copyright Roman Simiński Strona : 37

Dynamiczny przydział pamięci w języku C++, przykład sprzed lat Zasady przydziału pamięci w C++ są podobne. Występują jednak różnice: int main() int * p = 0; C ++ int main() int * p = NULL; ANSI C89 p = new int; p = malloc( sizeof( int ) ); if( p!= 0 ) *p = 10; cout << ++(*p); delete p; p = 0; if( p!= NULL ) *p = 10; cout << ++(*p); free( p ); p = NULL; W C++ wykorzystuje się wartość 0 jako wskaźnik pusty. Można korzystać ze stałej NULL, ale to wymaga włączenia odpowiedniego pliku nagłówkowego (np. stdlib). Copyright Roman Simiński Strona : 38

Dynamiczny przydział pamięci w języku C++, przykład na czasie Aktualnie, aby new oddało wartość 0, należy użyć jego specjalnej wersji: int main() int * p = 0; Stare dzieje w C ++ int main() int * p = 0; Aktualnie w C ++ p = new int; p = new (nothrow) int; if( p!= 0 ) *p = 10; cout << ++(*p); delete p; p = 0; if( p!= 0 ) *p = 10; cout << ++(*p); delete p; p = 0; Jeżeli nie użyjemy operatora w wersji new (nothrow), wygenerowany zostanie wyjątek bad_alloc. Copyright Roman Simiński Strona : 39

Dynamiczny przydział pamięci w języku C++, wyjątki Z wyjątkami współpracują instrukcje try i catch: int main() try int * p = new int; Aktualnie w C ++ *p = 10; cout << ++(*p); delete p; p = 0; catch( bad_alloc ) // Zrob cos gdy brak pamieci Mechanizm obsługi wyjątków oraz zasady stosowania try-catch zostaną omówione osobno. Copyright Roman Simiński Strona : 40