Programmazione Orientata agli Oggetti (OOP)

Da Bioingegneria Elettronica e Informatica.

Vitoantonio Bevilacqua vitoantonio.bevilacqua@poliba.it

Antonio Brunetti antonio.brunetti@poliba.it

Gianpaolo Francesco Trotta gianpaolofrancesco.trotta@poliba.it


Introduzione

La programmazione orientata agli oggetti (OOP) consiste in un paradigma di programmazione basato su modelli chiamati classi. Gli oggetti sono istanze di classi, e le classi sono messe in relazione fra loro attraverso i principi della ereditarietà (relazione "is a"), della aggregazione e della composizione (relazioni "has a").

La programmazione orientata agli oggetti supporta tre meccanismi (incapsulamento, ereditarietà e polimorfismo):

  1. Incapsulamento – si basa sul principio dell’information hiding (attributi o variabili e metodi o funzioni) e permette di nascondere all’utente i dettagli implementativi di una classe; quindi l’utente interagisce con gli oggetti di una determinata classe attraverso le interfacce messe a disposizione dall’oggetto stesso.
  2. Ereditarietà – la possibilità per una classe derivata di acquisire le caratteristiche di un’altra classe base; tramite le relazioni di generalizzazione/specializzazione, una superclasse definisce un concetto generale, che viene poi specializzato dalle sottoclassi.
  3. Polimorfismo – la possibilità di utilizzare uno stesso nome per definire metodi diversi permettendo di avere accesso a diverse implementazioni di una funzione attraverso un solo nome (un’interfaccia, più metodi). Il Polimorfismo può essere a compile time e a run time: il polimorfismo a compile time è noto come overloading, quello a run time come overridding.

Le Classi

Alla base della programmazione ad oggetti c'è il concetto di "classe". Una classe è un costrutto utilizzato per la modellazione di entità, quindi per la creazione (istanziazione) di oggetti; quindi, un'istanza della classe si definisce “oggetto”. All'interno di una classe si possono distinguere i seguenti elementi:

  • Attributo/i;
  • Metodo/i.

Gli "attributi", o variabili membro, definiti all'interno di una classe, costituiscono lo stato della classe stessa. Un "metodo" di una classe, invece, definisce una funzionalità della classe; i metodi servono a specificare il comportamento dell’entità (ovvero l’oggetto) di cui la classe è rappresentazione. Trattandosi di funzioni, un metodo può ritornare valori, oppure nulla (come le procedure o “funzioni void”).

L'esempio seguente mostra come è possibile implementare la classe Persona. In questo contesto, essa è caratterizzata da due attributi di tipo stringa (nome e cognome) e una variabile di tipo intero (id). Per quanto riguarda i metodi, la classe Persona è definita da 2 funzioni costruttore (una senza parametri, l'altra parametrizzata), una funzione per l'inserimento della variabile id, e una funzione per la stampa di nome e cognome.

  1. public class Persona 
  2. {
  3. 	private String nome;
  4. 	private String cognome;
  5. 	private int id;
  6.  
  7. 	public Persona() 
  8. 	{
  9. 		// Costruttore non parametrizzato della classe persona
  10. 	}
  11.  
  12. 	public Persona(String NOME, String COGNOME) 
  13. 	{
  14. 		nome = NOME;
  15. 		cognome = COGNOME;
  16. 	}
  17.  
  18. 	public void stampaNomeCognome() 
  19. 	{
  20. 		System.out.println(nome + " " + cognome);
  21. 	}
  22.  
  23. 	public void inserisciId(int ID) 
  24. 	{
  25. 		id = ID;
  26. 	}
  27. }

In qualunque punto del codice, è possibile istanziare oggetti di classi definite dall'utente. Per esempio, nel metodo main della classe Principale definita di seguito, è istanziato un oggetto di classe Persona (utilizzando il costruttore parametrizzato), e vengono richiamati i metodi definiti precedentemente.

  1. public class Principale 
  2. {
  3. 	public static void main(String[] args) 
  4. 	{
  5. 		Persona p = new Persona("Nome", "Cognome");
  6. 		p.inserisciId(1);
  7. 		p.stampaNomeCognome();
  8. 	}
  9. }

Ereditarietà

L'ereditarietà è il meccanismo per definire una nuova classe (classe derivata) come specializzazione di un’altra (classe base). Se la classe base modella un concetto generico, la classe derivata modella un concetto più specifico. Attraverso il principio dell'ereditarietà, la classe derivata:

  • dispone di tutte le funzionalità (attributi e metodi) di quella base;
  • può aggiungere funzionalità proprie;
  • può ridefinirne il funzionamento di metodi esistenti (polimorfismo).

In Java, si definisce una classe derivata attraverso la parola chiave “extends”, seguita dal nome della classe base.

Polimorfismo

Nell’ambito dei linguaggi di programmazione, il polimorfismo si riferisce in generale alla possibilità data ad una determinata espressione di assumere valori diversi in relazione ai tipi di dato a cui viene applicata. Il polimorfismo può essere implementato in modi diversi; i principali sono:

  • Overloading delle funzioni (method overloading): permette di "ridefinire" un medesimo metodo per set di parametri diversi (stesso nome, ma dominio diverso).
  • Ridefinizioni dei metodi ereditati (method overriding): in una sottoclasse è possibile modificare la definizione di un metodo presente nella superclasse facendo in modo tale che lo stesso metodo si comporti diversamente a seconda del tipo di oggetto su cui è invocato.

Esempio

Il codice seguente è quello relativo alla classe Persona. Come si può vedere, tale classe contiene un'unica variabile membro, una funzione costruttore (e la sua versione overloadata), e la due ulteriori funzioni.

  1. public class Persona {
  2. 	public int idp;
  3.  
  4. 	public Persona() {
  5. 		System.out.println("Costruttore senza parametri della classe Persona");
  6. 	}
  7.  
  8. 	public Persona(int IDP) {
  9. 		idp = IDP;
  10. 		System.out.println("Costruttore con parametri della classe Persona");
  11. 	}
  12.  
  13. 	public void stampaIDP() {
  14. 		System.out.println("Il valore della variabile ID di persona vale " + idp);
  15. 	}
  16.  
  17. 	public void stampaIDP(int notused) {
  18. 		System.out.println("Il valore della variabile ID di persona vale " + notused);
  19. 	}
  20.  
  21.       // Funzione che calcola la dimensione in Byte di una variabile intera
  22.       public int stampaDimensioneIntero() {
  23. 		int size = (Integer.SIZE) / Byte.SIZE;
  24.  
  25. 		return size;
  26. 	}
  27. }

Nelle seguenti linee di codice viene definita la classe Studente, derivata dalla classe persona, attraverso la parola chiave extends.

  1. public class Studente extends Persona {
  2. 	public int ids;
  3.  
  4. 	public Studente() {
  5. 		System.out.println("Costruttore senza parametri della classe Studente");
  6. 	}
  7.  
  8. 	public Studente(int IDS) {
  9.                 // Richiamo il costruttore parametrizzato della classe base
  10. 		super(IDS+1);
  11.  
  12.                  // Inizializzo la variabile membro della classe Studente
  13. 		ids = IDS;
  14. 		System.out.println("Costruttore con parametri della classe Studente");
  15. 	}
  16.  
  17. 	public void stampaIDs()
  18. 	{
  19. 		System.out.println("Valore dello studente stampato a " + ids);
  20. 	}
  21. }

Infine, nella classe principale vengono istanziati oggetti delle classi precedentemente definite, e richiamate le funzioni dichiarate e definite.

  1. public class Principale {
  2.  
  3. 	public static void main(String[] args) {
  4.  
  5. 		Persona p = new Persona(3);
  6. 		p.stampaIDP();
  7.  
  8. 		int dim = p.stampaDimensioneIntero();
  9. 		System.out.println("La dimensione dell'intero vale: " + dim);
  10.  
  11. 		Studente s = new Studente(5);
  12. 		s.stampaIDP();
  13. 	}
  14. }