Esercitazioni LPS: differenze tra le versioni
Da Bioingegneria Elettronica e Informatica.
(→Funzioni virtuali) |
(→Funzioni virtuali pure e Classi Astratte) |
||
Riga 191: | Riga 191: | ||
<syntaxhighlight lang="cpp" line> | <syntaxhighlight lang="cpp" line> | ||
#include <iostream> | #include <iostream> | ||
+ | #include <cmath> | ||
using namespace std; | using namespace std; | ||
+ | |||
+ | class Figura | ||
+ | { | ||
+ | // Classe Astratta: contiene almeno una funzione virtuale pura | ||
+ | protected: | ||
+ | double perimetro; | ||
+ | double area; | ||
+ | public: | ||
+ | Figura() { perimetro = 0; area = 0; } | ||
+ | |||
+ | virtual void calcolaPerimetro() = 0; // Funzione virtuale pura | ||
+ | virtual void calcolaArea() = 0; // Necessario ridefinire i metodi virtuali puri nelle classi derivati, altrimenti errore in compilazione | ||
+ | }; | ||
+ | |||
+ | class Triangolo : public Figura | ||
+ | { | ||
+ | private: | ||
+ | // Dimensione dei tre lati | ||
+ | double l1, l2, l3; | ||
+ | public: | ||
+ | Triangolo(double L1, double L2, double L3) { | ||
+ | l1 = L1; l2 = L2; l3 = L3; | ||
+ | } | ||
+ | |||
+ | void calcolaPerimetro() { | ||
+ | perimetro = l1 + l2 + l3; | ||
+ | cout << "Il perimetro del Triangolo con lati " << | ||
+ | l1 << " " << l2 << " e " << l3 << " vale: " << perimetro << endl; | ||
+ | } | ||
+ | |||
+ | // Calcolo area con formula di Erone | ||
+ | void calcolaArea() { | ||
+ | double p = (l1 + l2 + l3) / 2; | ||
+ | area = sqrt(p*(p - l1)*(p - l2)*(p - l3)); | ||
+ | cout << "L'area del Triangolo con lati " << | ||
+ | l1 << " " << l2 << " e " << l3 << " vale: " << area << endl; | ||
+ | } | ||
+ | }; | ||
+ | |||
+ | class Rettangolo : public Figura | ||
+ | { | ||
+ | private: | ||
+ | // Dimensione di base e altezza | ||
+ | double l1, l2; | ||
+ | public: | ||
+ | Rettangolo(double L1, double L2) { | ||
+ | l1 = L1; l2 = L2; | ||
+ | } | ||
+ | void calcolaPerimetro() { | ||
+ | perimetro = 2*(l1 + l2); | ||
+ | cout << "Il perimetro del Rettangolo con lati " << l1 << " e " << l2 << " vale: " << perimetro << endl; | ||
+ | } | ||
+ | |||
+ | void calcolaArea() { | ||
+ | area = l1 * l2; | ||
+ | cout << "L'area del Rettangolo con lati " << l1 << " e " << l2 << " vale: " << area << endl; | ||
+ | } | ||
+ | }; | ||
+ | |||
+ | class Cerchio : public Figura | ||
+ | { | ||
+ | private: | ||
+ | // Dimensione del raggio | ||
+ | double l1; | ||
+ | public: | ||
+ | Cerchio(double L1) { | ||
+ | l1 = L1; | ||
+ | } | ||
+ | void calcolaPerimetro() { | ||
+ | perimetro = 3.14 * 2 * l1; | ||
+ | cout << "La circonferenza del cerchio con raggio " << l1 << " vale: " << perimetro << endl; | ||
+ | } | ||
+ | |||
+ | void calcolaArea() { | ||
+ | area = 3.14 * l1 * l1; | ||
+ | cout << "L'area del cerchio con raggio " << l1 << " vale: " << area << endl; | ||
+ | } | ||
+ | }; | ||
+ | |||
int main() | int main() | ||
{ | { | ||
+ | Figura f; // Errore: non è consentito istanziare oggetti di tipo classe astratta | ||
+ | Figura *f1; | ||
+ | Triangolo t1(10.3, 9, 18.3); | ||
+ | Rettangolo r1(10, 3.5); | ||
+ | Cerchio c1(10); | ||
+ | |||
+ | |||
+ | f1 = &t1; | ||
+ | f1->calcolaPerimetro(); | ||
+ | f1->calcolaArea(); | ||
+ | |||
+ | f1 = &r1; | ||
+ | f1->calcolaPerimetro(); | ||
+ | f1->calcolaArea(); | ||
+ | |||
+ | f1 = &c1; | ||
+ | f1->calcolaPerimetro(); | ||
+ | f1->calcolaArea(); | ||
+ | system("pause"); | ||
+ | return 0; | ||
} | } | ||
</syntaxhighlight> | </syntaxhighlight> |
Versione delle 09:50, 22 mag 2019
Indice
Esercitazione del 2 e 15 maggio 2019
// questo è un commento a linea singola in puro stile C++
// la programmazione ad oggetti supporta //
// l'incapsulamento
// il polimorfismo
// l'ereditarietà
// esempio di incapsulamento: dati e funzioni in un oggetto
// esempio di polimorfismo a compile time: overloading di funzioni
// esempio di ereditarietà: classe persona e classe studente
// esempio di costruttore per inizializzare variabili a valori di default
// un oggetto è una istanza di una classe
// include la necessaria intestazione <iostream>
#include <iostream>
// l'istruzione using informa il compilatore che intende utilizzare il namespace standard
using namespace std;
// i namespace creano delle regioni di dichiarazione
// nel namespace standard viene dichiarata l'intera libreria standard del C++
// Creazione della classe persona
// una classe può contenere attributi (variabili) e metodi (funzioni) privati, protetti e pubblici
// per ora prevediamo soltanto attributi privati e metodi pubblici
class persona {
private: //private è pleonastico (si può omettere)
int IDP;
// IDP in quanto variabile privata non risulta visibile da funzioni che non siano metodi della classe persona
public:
void leggiIDP();
void leggiIDP(int);
void stampaIDP();
// segue la funzione costruttore, omonima della classe, e nella cui dichiarazione non si specifica il valore restituito
persona();
// segue la funzione costruttore overloadata con un parametro
persona(int);
// segue la funzione distruttore, omonima della classe ma preceduta dal TILDE, e nella cui dichiarazione non si specifica il valore restituito
~persona();
};
class studente : public persona {
private:
int IDS;
public:
void leggiIDS();
void stampaIPDS();
studente();
~studente();
};
void persona::leggiIDP()
{
cin >> IDP;
}
void persona::leggiIDP(int A)
{
IDP = A;
}
void persona::stampaIDP()
{
cout << IDP << endl;
// endl per andare a capo
}
persona::persona()
{
cout << "Persona costruita " << endl;
IDP = 0;
}
persona::persona(int A)
{
cout << "Persona costruita con parametro passato per valore " << endl;
IDP = A;
}
persona::~persona()
{
cout << "Persona distrutta " << endl;
}
studente::studente()
{
cout << "Studente costruito " << endl;
IDS = 0;
}
studente::~studente()
{
cout << "Studente distrutto " << endl;
}
void studente::leggiIDS()
{
cin >> IDS;
}
void studente::stampaIPDS()
{
cout << "Identificativo studente pari a " << IDS << endl;
}
// verificare il sorgente modificato ...
int main()
{
persona P, Q(3);
studente S; // verifica della chiamata del costruttore di persona senza parametri
// P.IDP=2; errore cannot access private member
P.stampaIDP();
P.leggiIDP();
P.stampaIDP();
Q.stampaIDP();
Q.leggiIDP(2);
Q.stampaIDP();
S.leggiIDP();
S.stampaIDP();
S.leggiIDS();
S.stampaIPDS();
// Decommentare su visual studio
// system("pause");
return 0;
}
Funzioni virtuali
#include <iostream>
using namespace std;
class Base
{
public:
virtual void stampa() {
cout << "Metodo stampa della classe Base" << endl;
}
};
class DerivataUno : public Base
{
public:
void stampa() {
cout << "Metodo stampa della classe Derivata1" << endl;
}
};
class DerivataDue : public Base
{
public:
void funzione() {
cout << "Funzione specifica della classe DerivataDue" << endl;
}
};
int main()
{
Base b, *p;
DerivataUno d1;
DerivataDue d2;
p = &b;
// Richiamata la funzione stampa della classe Base
p->stampa();
p = &d1;
// Richiamata la funzione stampa della classe DerivataUno
p->stampa();
p = &d2;
// Richiamata la funzione stampa della classe Base
p->stampa();
// Richiama il metodo specifico della classe DerivataDue ->
// -> cast a punatatore di tipo classe derivata
((DerivataDue*)p)->funzione();
system("pause");
}
Funzioni virtuali pure e Classi Astratte
#include <iostream>
#include <cmath>
using namespace std;
class Figura
{
// Classe Astratta: contiene almeno una funzione virtuale pura
protected:
double perimetro;
double area;
public:
Figura() { perimetro = 0; area = 0; }
virtual void calcolaPerimetro() = 0; // Funzione virtuale pura
virtual void calcolaArea() = 0; // Necessario ridefinire i metodi virtuali puri nelle classi derivati, altrimenti errore in compilazione
};
class Triangolo : public Figura
{
private:
// Dimensione dei tre lati
double l1, l2, l3;
public:
Triangolo(double L1, double L2, double L3) {
l1 = L1; l2 = L2; l3 = L3;
}
void calcolaPerimetro() {
perimetro = l1 + l2 + l3;
cout << "Il perimetro del Triangolo con lati " <<
l1 << " " << l2 << " e " << l3 << " vale: " << perimetro << endl;
}
// Calcolo area con formula di Erone
void calcolaArea() {
double p = (l1 + l2 + l3) / 2;
area = sqrt(p*(p - l1)*(p - l2)*(p - l3));
cout << "L'area del Triangolo con lati " <<
l1 << " " << l2 << " e " << l3 << " vale: " << area << endl;
}
};
class Rettangolo : public Figura
{
private:
// Dimensione di base e altezza
double l1, l2;
public:
Rettangolo(double L1, double L2) {
l1 = L1; l2 = L2;
}
void calcolaPerimetro() {
perimetro = 2*(l1 + l2);
cout << "Il perimetro del Rettangolo con lati " << l1 << " e " << l2 << " vale: " << perimetro << endl;
}
void calcolaArea() {
area = l1 * l2;
cout << "L'area del Rettangolo con lati " << l1 << " e " << l2 << " vale: " << area << endl;
}
};
class Cerchio : public Figura
{
private:
// Dimensione del raggio
double l1;
public:
Cerchio(double L1) {
l1 = L1;
}
void calcolaPerimetro() {
perimetro = 3.14 * 2 * l1;
cout << "La circonferenza del cerchio con raggio " << l1 << " vale: " << perimetro << endl;
}
void calcolaArea() {
area = 3.14 * l1 * l1;
cout << "L'area del cerchio con raggio " << l1 << " vale: " << area << endl;
}
};
int main()
{
Figura f; // Errore: non è consentito istanziare oggetti di tipo classe astratta
Figura *f1;
Triangolo t1(10.3, 9, 18.3);
Rettangolo r1(10, 3.5);
Cerchio c1(10);
f1 = &t1;
f1->calcolaPerimetro();
f1->calcolaArea();
f1 = &r1;
f1->calcolaPerimetro();
f1->calcolaArea();
f1 = &c1;
f1->calcolaPerimetro();
f1->calcolaArea();
system("pause");
return 0;
}