Iznimke C++ nudi mehanizam izbacivanja i hvatanja iznimaka za postupanje s greškama pri izvršavanju programa. Uporaba tog mehanizma nije nužna jer se.

Slides:



Advertisements
Similar presentations
C++ Exceptions STL Vector. Example int Quotient (int numer, int denom} { if (denom != 0) return (numer/denom); else //What to do?? }
Advertisements

Prof: doc.dr. Samir Lemeš student: Samir Hrnjić. System restore je komponenta Microsoftovih operativnih sistema Windows Serveri ne podržavaju opciju System.
Mark Redekopp David Kempe
Petlje WHILE – WEND.
Mrezno racunarstvo Java, niti.
RP3/predavanje08 Ugniježdeni tipovi Iznimke 10/11/2018
Računarski praktikum 3 Vježbe 07
Java Hello world !.
2.3 Testing Programs 12 – Program Correctness.
Algoritamske/programske strukture
Nadgradnja klasa i nasljeđivanje – 3
Programi,Podaci,Varijable,Računanje- Uvod
Objektno programiranje (C++)
Programiranje - Blokovi naredbi i logički tipovi –
OSNOVE PROGRAMIRANJA U PROGRAMSKOM JEZIKU
Java Petlje i logika - 1.
CheckBox RadioButton RadioGroup
Naredbe ciklusa.
Petlje FOR - NEXT.
REPEAT…UNTIL Naredbe ciklusa.
Programi,Podaci,Varijable,Računanje - 2
Java Iznimke - exceptions.
Објектно орјентисано програмирање
KREIRANJE OBJEKATA.
Java Klase (Classes).
Uvod u programiranje - matematika – X predavanje
Arrays and strings -2 (nizovi i znakovni nizovi)
Throwing exceptions.
14 UNUTRAŠNJE I ANONIMNE KLASE
Изведене класе Вишеструко извођење Полиморфизам
Elementi programskog jezika PASCAL
Arrays and strings -1 (nizovi i znakovni nizovi)
Klasa grafik.
Struktura MAC adrese i Ethernet okvira
Programiranje II Obrada izuzetaka.
Naredba Case Višestruko grananje.
Radost prijateljstva.
M-datoteke.
MessageBox.
Objektno orijentisano programiranje
(カックロ ) Ivo Ivanišević Ena Melvan
Ključne reči,identifikatori, konstante i promenljive
Internet FTP usluga.
PROGRAMSKI JEZIK PASCAL
Visual Basic – Prvi primjer
Osnovni simboli jezika Pascal
Do While ... Loop struktura
Default konstruktor struct C{ int i; }; struct D{ D( ):i(1){ }
Programiranje – Small Basic
Throwing exceptions.
CSC 270 – Survey of Programming Languages
Programiranje - Uvod - Kolegij: Programski jezik C++
Programiranje za Internet
Skup instrukcija procesora
Fakultet elektrotehnike i računarstva
Naredbe u php-u.
Programski jezik C++ - Vježbe - 5. dio
Programiranje - Naredbe za kontrolu toka programa – 1. dio
Programiranje - Naredbe za kontrolu toka programa – 3. dio
Programski jezik C++ - Vježbe - 1. dio
Radost prijateljstva.
Ponavljanje Pisana provjera
Programiranje - Naredbe za kontrolu toka programa – 1. dio
INTERPOLACIJA PO DIJELOVIMA POLINOMIMA
Throwing exceptions.
Vježbenica 2: struktura grananja – 2.dio
Hour of Code Coding Jetpack Jumper Mateja Hržica, MSP.
Uvjetne petlje.
Višestruko grananje.
Programski jezik C++ - Vježbe - 2. dio
Presentation transcript:

Iznimke C++ nudi mehanizam izbacivanja i hvatanja iznimaka za postupanje s greškama pri izvršavanju programa. Uporaba tog mehanizma nije nužna jer se greške mogu tretirati na tradicionalan način, ali on nudi važnu mogućnost separacije koda koji reagira na grešku od koda u kojem je došlo do greške. To je važno u onim situacijama kada na grešku nije moguće adekvatno reagirati tamo gdje se pojavila.

Iznimke Dio programa koji može izbaciti izuzetak zatvori se u try blok. Njega slijede jedan ili više catch blokova koji hvataju i procesuiraju izuzetak. Izuzetak se izbacuje pomoću throw naredbe. Izvršavanje programa se zaustavlja u liniji koda u kojoj je izbačen izuzetak i počinje potraga za prvim catch blokom koji deklarira izbačeni izuzetak. Ako je takav blok nađen, program se nastavlja u njemu. Ako u try bloku nije izbačen izuzetak, pripadni catch blokovi se preskaču i program se nastavlja iz njih. Ako je izuzetak izbačen, ali nije nađen catch blok koji ga deklarira (premda može biti drugih catch blokova) potraga za catch blokom se nastavlja u pozivnom programu, rekurzivno, sve dok se ne nađe. Ako nije nađen niti u main funkciji, program će biti zaustavljen.

Iznimke Try-catch blok oblika try { f(); } catch (tipIznimke &e) { …….      } mogao bi se realizirati i pomoću if-else uvjeta (od funkcije očekujemo da vraća povratnu vrijednost pomoću koje možemo kontrolirati greške). Npr. možemo pisati:

Iznimke int r = f( ); if (!r) { ... } else { . } gdje funkcija f vraća nulu ako nije došlo do pogreške, odnosno neku nenul vrijednost u slučaju pogreške.

Iznimke Takav način tretiranja pogrešaka je vrlo nepraktičan, posebno ako poziv funkcije f uključuje dodatne pozive funkcija. Sljedeći kod     try {    …  f( );    …    }  catch (tipIznimke& e) { … }         void f( )  {  … f1( );  … }  void f1( ) { …  f2( );  …  }  …….. void f6( )  { …   f7( );  …  }    void f7( )  {    …      throw iznimka;     … } bi bez try-catch bloka mogao izgledati ovako:

Iznimke int f(){ …… int r=f1(); if (!r) { …… } ……. else { …. } return 0; } int f1(){ …. int r=f2(); if (r) return 1; …… ………. int f7(){ ….. if (pogreška) return 1; …..

Iznimke int main(){ throw 1000; // throw izvan try bloka, nema catch bloka koji može uhvatiti iznimku try { throw 1000; } catch ( int e ){ cout<< " Uhvacena int iznimka. "; return 0;

Iznimke int main(){ try{ throw 1; throw 2; // ne izvršava se } catch (int e) { cout<<"Uhvatio"<<e; return 0;

Iznimke int main(){ try{ int a=2; throw a; } catch (int e){ cout<<"Izbacen je "<<a; // varijabla više ne postoji cout<<"Uhvacen je "<<e; return 0;

Iznimke #include <vector> #include <iostream> int main() {    std::vector<double> v(2,1.0); // vektor od dva elementa inicijalizirana s 1.0      std::cout << v.at(3) << std::endl; // izbacuje izuzetak std::out_of_range      return 0; }

Iznimke #include <vector> #include <iostream> void f(int i, std::vector<double>& v) {     std::cout << v.at(i) << std::endl; // izbacuje izuzetak ovisno o indeksu i      throw 1000; // izbacuje izuzetak      v[i] = 3.14; // nikad neće biti izvršeno } int main() {      std::vector<double> v(2,1.0);      int i;      std::cin >> i;      try{    f(i,v);  }      catch(int e)     {          std::cout << "Uhvatio cjelobrojni izuzetak." << std::endl;      }      std::cout << "Zavrsio." << std::endl;      return 0;

Iznimke #include <vector> #include <iostream> #include <stdexcept> int main() {      std::vector<double> v(2,1.0);      int i;      std::cin >> i;      try{         f(i,v);     }      catch(int e)     {          std::cout << "Uhvatio cjelobrojni izuzetak." <<std::endl;      }      catch(std::out_of_range e)     {            std::cout << "Uhvatio out_of_range izuzetak." << std::endl;            std::cout << e.what() << std::endl;      }      std::cout << "Zavrsio." << std::endl;      return 0; }

Iznimke int main( ){ try{ throw 1; } catch (string e){ cout<<"Uhvacen je string"; catch(...) { // ovaj blok hvata bilo koju iznimku cout<<"Uhvacena je neka iznimka"; return 0;

Iznimke Standardne iznimke: exception runtime_error range_error overflow_error underflow_error logic_error domain_error invalid_argument length_error. bad_alloc bad_cast Sve ove klase nude metodu what koja vraća string koji opisuje izuzetak.

Iznimke class exception { public: exception(); exception(const char * const &message); exception(const char * const &message, int); exception(const exception &right); exception& operator=(const exception &right); virtual ~exception(); virtual const char *what() const; };

Iznimke class logic_error : public exception { public: explicit logic_error(const string& message); virtual const char *what(); }; class runtime_error : public exception { explicit runtime_error(const string& message);

Iznimke Sljedeće klase su izvedene iz klase logic_error: domain_error invalid_argument length_error out_of_range Iz klase runtime_error su izvedene: overflow_error range_error underflow_error

Iznimke void f(){ try { throw std::string("u funkciji f"); } catch(int e) { std::cout<<"Uhvatio int iznimku - u funkciji f"; } } int main() { try f(); catch(int e) std::cout<<"Uhvatio sam int iznimku – u funkciji main"; catch(std::string e) std::cout<<"Uhvatio sam string iznimku";

Iznimke void f(){ try { throw 1000; } catch(int e) { std::cout<<"Uhvatio int iznimku - u funkciji f"; } } int main() { try f(); catch(int e) std::cout<<"Uhvatio sam int iznimku – u funkciji main"; catch(std::string e) std::cout<<"Uhvatio sam string iznimku";

Iznimke Funkcijski try-catch blok #include <iostream> #include <stdexcept> using namespace std; void f(int i) try{ if (i<3) throw out_of_range ("Broj treba biti veci od 2."); cout<<i<<endl; } catch(out_of_range e) { std::cout<<e.what()<<endl; int main() { f(2); f(3); return 0; Funkcijski try-catch blok

Iznimke Catch blok koji obrađuje iznimku je prvi u nizu pripadnih catch blokova koji može primiti tip iznimke. Pritom nisu dopuštene sve konverzije koje su dopuštene pri prosljeđivanju argumenata funkcijama. Dozvoljene su sljedeće: Konverzija iz nekonstantnog objekta u konstantan je dozvoljena. Specijalno, izbacivanje nonconst objekta se slaže s catch blokom koji prima const referencu. Konverzija iz izvedenog u bazni tip je dozvoljena (detalji kasnije). Polja i funkcije se konvertiraju u odgovarajuće pokazivače. Najspecijaliziraniji catch blok bi trebao biti prvi u nizu.

Iznimke int main() { try{ int a=2; throw a; } catch (const double& ){cout<<"Uhvatio double"; } catch (const int& ){cout<<"Uhvatio int"; } // ovaj catch hvata iznimku return 0;

Iznimke void f(string s) { std::cout<<"U funkciji sam f"; } int main() try f("Tekst"); // konverzija kod funkcije radi throw "Tekst"; catch(string) { std::cout<<"Uhvatio sam string"; } catch(exception) { std::cout<<"Uhvatio sam objekt tipa exception"; } // nema catch bloka koji bi uhvatio iznimku return 0;

Iznimke void f(string s) { std::cout<<"U funkciji sam f"; } int main() try f("Tekst"); string s(“Tekst”); throw s; catch(string) { std::cout<<"Uhvatio sam string"; } // ovaj catch sad može uhvatiti iznimku catch(exception) { std::cout<<"Uhvatio sam exception"; } return 0;

Iznimke void f( int & i ) { cout<<"f( int & )"; } void f(const int & i) { cout<<"f( int & )"; } // ove funkcije mogu koegzistirati int main( ) { int a=2; f(a); // f(int &) return 0; }

Iznimke void f( int & i ) { cout<<"f( int & )"; } void f(const int & i) { cout<<"f( int & )"; } int main( ) { int a=2; f(a); try{ throw a; } catch (const int & e){ cout<<"Uhvatio "<<e<<" preko const reference"; catch (int & e){ cout<<"Uhvatio "<<e<<" preko nonconst reference"; return 0;

Iznimke void f( int & i ) { cout<<"f( int & )"; } void f(const int & i) { cout<<"f( int & )"; } int main( ) { int a=2; f(a); try{ throw a; } catch (const int & e){ cout<<"Uhvatio "<<e<<" preko const reference"; catch (int & e){ cout<<"Uhvatio "<<e<<" preko nonconst reference"; // treba izbaciti jedan od catch blokova return 0;

Iznimke void f( ); void f1( ); void f2( ); void f3( ); void f( ) { f1( ); } void f1( ) { try { f2( ); } catch (const char * e) { cout<<"f1"; } } void f2( ) { try { f3 ( ); } catch (string e) { cout<<"f2" ; } } void f3( ) { throw "Iznimka"; } int main( ) { try { f( ); } catch (const char * e) { cout<<"main"; } // ispisuje f1 return 0; }

Iznimke Ponekad catch blok koji je uhvatio iznimku nije u mogućnosti u potpunosti obraditi tu iznimku, već samo djelomično. Stoga, taj catch blok može ponovo izbaciti uhvaćenu iznimku koju može uhvatiti neki od odgovarajućih catch blokova koji se u lancu pripadnih funkcijskih poziva nalazi iznad trenutnog catch bloka. Ponovno izbacivanje iznimke postiže se opet pomoću ključne riječi throw, ali ovaj put bez navođenja objekta, odnosno izraza: throw;

Iznimke void f( ); void f1( ); void f2( ); void f3( ); void f( ) { f1( ); } void f1( ) { try { f2( ); } catch (const char * e) { cout<<"f1 "; throw; } } void f2( ) { try { f3 ( ); } catch (string e) { cout<<"f2 " ; } } void f3( ) { throw "Iznimka"; } int main( ) { try { f( ); } catch (const char * e) { cout<<"main"; } cout<<“Sve u redu”; // f1 main return 0; }

Iznimke Objekt koji se time izbacuje je originalan objekt koji je uhvaćen, a ne catch parametar. Dakle, objekt kojeg izbacuje catch blok se može razlikovati od objekta kojeg je taj catch blok uhvatio jedino ako je hvatanje objekta izvršeno po referenci i ako je catch blok prethodno mijenjao taj objekt.

Iznimke void f1( ); void f( ){ try{ f1( ); } catch(int &e){ ++e; throw; } } void f1( ) { try { throw 1; } catch (int e){ ++e; throw; } int main(){ try{ f(); } catch(int e){ cout<<"Uhvacen je broj"<<e; } return 0;

Iznimke void f1( ); void f( ){ try{ f1( ); } catch(int &e){ ++e; throw; } i // izbacuje 2 } void f1( ) { try { throw 1; } catch (int e){ ++e; throw; } // izbacuje 1 int main(){ try{ f(); } catch(int e){ cout<<"Uhvacen je broj"<<e; } return 0;

Iznimke void f1( ); void f( ){ try{ f1( ); } catch(int &e){ ++e; throw e; } } void f1( ) { try { throw 1; } catch (int e){ ++e; throw e; } int main(){ try{ f(); } catch(int e){ cout<<"Uhvacen je broj"<<e; } return 0;

Iznimke void f1( ); void f( ){ try{ f1( ); } catch(int &e){ ++e; throw e; } // izbacuje 3 } void f1( ) { try { throw 1; } catch (int e){ ++e; throw e; } // izbacuje 2 int main(){ try{ f(); } catch(int e){ cout<<"Uhvacen je broj"<<e; } return 0;