Esercitazioni LPS: differenze tra le versioni
Da Bioingegneria Elettronica e Informatica.
Riga 301: | Riga 301: | ||
− | == | + | == Esempi di costruttore di copie== |
<syntaxhighlight lang="cpp" line> | <syntaxhighlight lang="cpp" line> | ||
#include <iostream> | #include <iostream> | ||
− | #include < | + | #include <new> |
− | + | #include <cstdlib> | |
using namespace std; | using namespace std; | ||
− | + | // ATTENZIONE: in ambiente Visual Studio avviare senza eseguire Debug (CTRL+F5) | |
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | // Terzo esempio per creare costruttori di copie | |
− | + | // N.B. quando si esegue una copia i due oggetti sono identici bit a bit, | |
− | + | // quando però la inizializzazione comporta l'allocazione di memoria non deve | |
+ | // essere fatta una copia bit a bit | ||
− | + | // Questo programma crea una classe per array "sicuri" | |
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | |||
− | class | + | class vettore |
{ | { | ||
− | + | int *p; | |
− | + | int dimensione; | |
− | + | ||
public: | public: | ||
− | + | vettore(int dim) | |
− | + | { | |
− | + | cout << "costruzione" << endl; | |
− | + | p = new int[dim]; | |
− | + | dimensione = dim; | |
− | cout << " | + | |
} | } | ||
− | + | ~vettore() { delete[] p; cout << "distruzione" << endl; } | |
− | + | // costruttore di copie | |
− | + | vettore(const vettore &vett); | |
− | + | ||
− | + | ||
− | + | void imposta(int i, int j) | |
− | + | { | |
− | + | if (i >= 0 && i <dimensione) | |
− | + | p[i] = j; | |
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
− | + | ||
} | } | ||
− | + | int prendi(int i) | |
− | + | { | |
− | + | return p[i]; | |
} | } | ||
}; | }; | ||
+ | // Costruttore di copie | ||
− | + | vettore::vettore(const vettore &vett) | |
{ | { | ||
− | + | int i; | |
− | + | p = new int[vett.dimensione]; | |
− | + | cout << "costruttore di copie" << endl; | |
− | + | for (i = 0; i<vett.dimensione; i++) | |
− | + | p[i] = vett.p[i]; | |
+ | } | ||
− | + | int main() | |
− | + | { | |
− | + | vettore num(10); | |
+ | int i; | ||
+ | for (i = 0; i<10; i++) num.imposta(i, i); | ||
+ | for (i = 9; i >= 0; i--) cout << num.prendi(i); | ||
+ | cout << "\n"; | ||
− | + | // i vettori num e x contengono gli stessi valori ma ciascun vettore | |
− | + | // si trova in aree diverse di memoria | |
− | + | ||
− | + | // crea un altro vettore e lo inizializza con num | |
− | + | vettore x(num); // richiama il costruttore di copie; | |
− | + | for (i = 0; i<10; i++) cout << num.prendi(i); | |
− | + | cout << "\nciao\n"; | |
− | + | ||
return 0; | return 0; | ||
} | } | ||
+ | |||
+ | /* | ||
+ | costruzione | ||
+ | 9876543210 | ||
+ | costruttore di copie | ||
+ | 0123456789 | ||
+ | ciao | ||
+ | distruzione | ||
+ | distruzione | ||
+ | Press any key to continue | ||
+ | */ | ||
</syntaxhighlight> | </syntaxhighlight> |
Versione delle 13:42, 29 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 DerivataUno" << 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;
}
Esempi di costruttore di copie
#include <iostream>
#include <new>
#include <cstdlib>
using namespace std;
// ATTENZIONE: in ambiente Visual Studio avviare senza eseguire Debug (CTRL+F5)
// Terzo esempio per creare costruttori di copie
// N.B. quando si esegue una copia i due oggetti sono identici bit a bit,
// quando però la inizializzazione comporta l'allocazione di memoria non deve
// essere fatta una copia bit a bit
// Questo programma crea una classe per array "sicuri"
class vettore
{
int *p;
int dimensione;
public:
vettore(int dim)
{
cout << "costruzione" << endl;
p = new int[dim];
dimensione = dim;
}
~vettore() { delete[] p; cout << "distruzione" << endl; }
// costruttore di copie
vettore(const vettore &vett);
void imposta(int i, int j)
{
if (i >= 0 && i <dimensione)
p[i] = j;
}
int prendi(int i)
{
return p[i];
}
};
// Costruttore di copie
vettore::vettore(const vettore &vett)
{
int i;
p = new int[vett.dimensione];
cout << "costruttore di copie" << endl;
for (i = 0; i<vett.dimensione; i++)
p[i] = vett.p[i];
}
int main()
{
vettore num(10);
int i;
for (i = 0; i<10; i++) num.imposta(i, i);
for (i = 9; i >= 0; i--) cout << num.prendi(i);
cout << "\n";
// i vettori num e x contengono gli stessi valori ma ciascun vettore
// si trova in aree diverse di memoria
// crea un altro vettore e lo inizializza con num
vettore x(num); // richiama il costruttore di copie;
for (i = 0; i<10; i++) cout << num.prendi(i);
cout << "\nciao\n";
return 0;
}
/*
costruzione
9876543210
costruttore di copie
0123456789
ciao
distruzione
distruzione
Press any key to continue
*/