L’obiettivo della programmazione a oggetti è quello di rendere il codice riutilizzabile, si basa sulla creazione di classi e oggetti. Le classi forniscono un mezzo per costruire nuove strutture dati, un oggetto è un’istanza di quella classe
Esempio:
Proviamo adesso a ripensare alle immagini (matrici) ma come oggetti, quindi con attributi e metodi.
- class Color indica la definizione della classe.
- __init__ è il costruttore ovvero il metodo che crea oggetti, in questo caso crea oggetti e memorizza i valori r, g, b.
- __self__ va aggiunto nei metodi di una classe. Si riferisce all’oggetto istanza che verrà creato successivamente quando chiamiamo la classe con c1 = Color(0,0,0), c1 sarebbe “self”.
- __repr__ è il metodo che riscriviamo per mostrare la rappresentazione a video dell’oggetto quando lo stampiamo con print()
- NB solitamente in Python gli attributi con _ sono da considerarsi privati (_r, _g, _b).
Ereditarietà
Cosa succede se abbiamo un colore con ‘alpha’ channel? Ovvero un colore che estende la tripla RGB codificando la trasparenza dei colori in un quarto canale alpha.
La classe ColorAlpha eredita tutti gli attributi e metodi dalla classe Color ma la estende con l’attributo alpha, vediamo come scriverlo in codice:
Tramite super chiamo il costruttore della classe madre Color e poi estendo l’oggetto con l’attributo alpha. Anche il metodo __repr__ è un estensione del metodo della classe madre. Quindi per creare un oggetto di tipo ColorAlpha dovrò scrivere:
Attributi di Classe e di Oggetto
- n_istances sarà un attributo di classe, ovvero in comune tra tutte le istanze
- r, g, b sono attributi di oggetto, quindi ogni oggetto avrà quegli attributi ma con valori diversi
- NB Posso accedere agli attributi di classe anche dagli oggetti
Le due righe di codice accedono allo stesso dato. Devo stare attento però a non scrivere un attributo di classe tramite un oggetto, in quel caso l’attributo diventa specifico per quell’istanza
Metodi di Classe
Un metodo di classe riceve una classe come primo argomento implicito, così come una istanza oggetto riceve un riferimento a se stessa.
@classmethod è un decoratore, ovvero serve ad aggiungere funzionalità ad una funzione senza modificarla.
Decoratori
Assumiamo di avere una funzione che non possiamo modificare, ma vogliamo aggiungere delle funzionalità ad essa, come facciamo?
Il trucco sta nel ricordarsi che possiamo passare funzioni ad altre funzioni
Eseguendo dec() ottengo:
Invece di assegnare ogni volta una variabile come nuova funzione, posso usare direttamente il mio decoratore con la seguente sintassi:
Overloading Operatori
Dopo aver creato una nuova struttura dati, magari vogliamo cambiare il suo comportamento quando effettuiamo operazioni tra sue istanze, come ad esempio la somma o la sottrazione. Lo abbiamo già fatto con il metodo __repr__ ma vediamo come fare:
In questo modo abbiamo effettuato l’overloading dell’operatore somma, andando quindi a sommare due istanze della classe Color otterremo un oggetto che ha come valori la somma dei valori RGB.
E’ possibile definire la somma fra due colori e anche la moltiplicazione fra colore e scalare?
In questo modo se nella moltiplicazione, i fattori sono due colori moltiplichiamo tra loro i canali RGB, altrimenti se abbiamo un colore e uno scalare, moltiplichiamo i canali RGB per quell’intero. Nel caso riceviamo un tipo diverso da int o Color solleviamo un’eccezione e spieghiamo cosa è successo.
Gestione delle eccezioni
Come abbiamo visto prima possiamo generare degli errori, ma possiamo anche gestirli per non far terminare il nostro programma. Utilizziamo il costrutto try - except - finally
- try: prova ad eseguire il blocco
- except: cattura l’eccezione specificata
- finally: assicura che il blocco di codice sia sempre eseguito
Possiamo anche gestire più eccezioni nello stesso blocco