Sandalo 655 1574 Pelle Pikolinos Marrone zq4n8B

Come in altri linguaggi ad esempio C++ o Java, anche in Swift esistono le enumerazioni. Questo particolare tipo di dato risulta estremamente utile quando si vuole rappresentare un insieme finito e bene definito di valori. Le enumerazioni in Swift dispongono di una comoda sintassi e sono di per sé immutabili. Vediamo subito la sintassi del costrutto:

1
2
gold 500 black Donna Balance Nero New Sneaker
3
enum SomeEnumeration {
// la definizione dell'enumerazione va inserita qui
}

Una semplice enumerazione di base che rappresenta i punti cardinali potrebbe essere dunque:

Peter Scarpe Kaiser Col 47921074 Tacco Blau Donna rqr4z
1
2
3
4
5
6
enum CompassPoint {
    case north
    case south
    case east
    case west
}

Come si può notare, l’enumerazione va per casi. Ogni caso viene rappresentato attraverso la keyword case succeduta da un identificatore.
In casi cosi semplici, in cui i vari casi non possiedono proprietà, i case multipli possono essere scritti su una sola riga come ad esmepio:

1
2
3
enum Planet {
    case mercury , venus , earth , mars , jupiter , saturn , uranus , neptune
}

Per creare invece una variabile enumeratore, la sintassi è semplicissima:

1
2
3
4
var directionToHead = CompassPoint. west

//cambia il valore della variabile da west a east
directionToHead = . east

Una volta istanziata una variabile del tipo enumeratore che ci interessa, attraverso la Type Inference è possibile cambiare il valore stesso apponendo un case-name preceduto dal punto come nell’esempio mostrato.

Enumerazioni e Matching

Attraverso il costrutto switch è possibile eseguire operazioni di matching su una variabile enumeratore in maniera estremamente semplice e pulita. Vediamo come:

Donna Donna Pleaser Pleaser Pleaser Donna Donna Pleaser Donna qrtqF
1
2
3
4
5
6
7
8
9
10
11
12
directionToHead = . south
switch directionToHead {
    case . north :
        print ( "Lots of planets have a north" )
    case . south :
        print ( "Watch out for penguins" )
    case . east :
        print ( "Where the sun rises" )
    case . west :
        print ( black Nero Balance gold Sneaker 500 New Donna "Where the skies are blue" )
}
// Stampa "Watch out for penguins"

Il meccanismo della Type Inference ci aiuta ancora una volta. Ovviamente è sempre possibile apporre un default nello switch, qualora non volessimo prevedere tutti i case.

Enumerazioni e valori

Finora abbiamo visto soltanto la definizione e l’impiego basilare delle enumerazioni. Vediamo ora come fare a definire dei valori associati ad ogni case mediante un esempio sul barcode. Ricordiamo che il barcode può essere espresso sia come qrCode o come codice a barre numerico.

Sneaker New Nero black Donna gold 500 Balance

A sinistra il qrCode, a destra il barcode numerico.

 

Se volessimo rappresentare un barcode dunque basterebbe una semplice enumerazione cosi strutturata:

1
2
3
4
5
6
7
8
9
enum Barcode {
    case upc ( Int , Int , Int , Int )
    case qrCode ( String )
}

//istanziazione esempio
var productBarcode = Barcode. upc ( 8 , 85909 , 51226 , 3 )
//modifica del barcode
productBarcode = . qrCode ( "ABCDEFGHIJKLMNOP" )

Come si può vedere basta passare al case i parametri richiesti, racchiudendo cosi i dati nell’enumeratore. Se ricordate bene lo switch permette di eseguire il matching di tuple, cosa che fa al caso nostro essendo proprio tuple costanti:

1
2
3
4
5
6
7
switch productBarcode {
    case . upc ( let numberSystem , let manufacturer , let product , Nero Balance black New gold Donna 500 Sneaker let check ) :
        print ( "UPC: \(numberSystem), \(manufacturer), \(product), \(check)." )
    case . qrCode ( let productCode ) :
        print ( "QR code: \(productCode)." )
}
// Stampa "QR code: ABCDEFGHIJKLMNOP."

E visto che in questo caso abbiamo apposto il let a tutti i parametri del case, basta inserirlo prima del case matching in modo da considerare la tupla integralmente:

1
2
3
4
5
6
7
switch productBarcode {
    case let . upc (numberSystem , manufacturer , product , check ) :
        print ( "UPC : \(numberSystem), \(manufacturer), \(product), \(check)." )
    case let . qrCode (productCode ) :
        print ( "QR code: \(productCode)." )
}
// Stampa "QR code: ABCDEFGHIJKLMNOP."

Un’enumerazione può avere dei valori di default per i vari casi. Per assegnarne uno bsogna dichiararne il tipo come estensione del tipo. Vediamo come si fa in pratica:

1
2
3
4
5
enum ASCIIControlCharacter : Character {
    case tab = "\t"
    case lineFeed = "\n"
    case carriageReturn = "\r"
}

Il codice ASCII “estende” il tipo carattere ed assegniamo un carattere di escape ad ogni case.
Essendo le enumerazioni delle rappresentazioni biunivoche con l’insieme dei numeri naturali secondo la teoria matematica, è possibile in Swift inserire un raw value di base intero. Questo valore iniziale apposto a un case, farà si che tutti i case successivi avranno un valore successivo a quello definito. Infatti secondo l’algebra induttiva, N è definito in base allo 0 e alla funzione succ(n) ovvero successore di un certo numero.

1
2
3
4
5
6
7
8
9
10
11
12
enum Planet : Int {
    case mercury = 1 , venus , earth , mars , jupiter , saturn , uranus , neptune
}

let earthsOrder = Planet. earth. rawValue
// earthsOrder é 3

let sunsetDirection = CompassPoint. west. rawValue
// sunsetDirection é "west"

let possiblePlanet = Planet (rawValue : 7 )
// possiblePlanet é di tipo Planet? ed uguale a Planet.uranus

Infatti jupiter sarà 5. Se avessimo apposto il raw value 1 a mars, jupiter sarebbe 2. Se si vuole estrapolare un enumeratore attraverso il raw value, si otterrà un Enum? perché effettivamente potrebbe essere nil e dunque non esistere. Una verifica per evitare il nil è utilizzare l’if let:

1
2
New gold Balance black Nero Donna Sneaker 500
3
4
gold Donna New 500 black Sneaker Nero Balance
5
6
7
8
9
10
11
12
let positionToFind = 11
if let somePlanet = Planet (rawValue : positionToFind ) {
    switch somePlanet {
        black Sneaker New 500 Balance Donna gold Nero case . earth :
            print ( "Mostly harmless" )
        default :Da Mulberry Scarpe 1623 On grey Trail Running Midtop Cloudventure qTqta
            print ( "Not a safe place for humans" )
    }
} else {
    print (Di Aperto Sandali Moda Verde Donna Caviglia Scarpe Piattaforma Espadrillas Da x6ZXqU "There isn't a planet at position \(positionToFind)" )
}
// Stampa "There isn't a planet at position 11"

Enumerazioni Ricorsive

Ogni linguaggio di programmazione funzionale possiede i tipi induttivi o algebrici, ovverosia, la possibilità di creare dei tipi di dato definiti attraverso funzioni costruttori, governati dalla teoria dei tipi funzionale. In Swift questo concetto è stato semplificato permettendo allo sviluppatore di definire un tipo algebrico attraverso l’enumerazione. Induzione e ricorsione sono legati in molti casi, per questo i case vengono trattati come componenti ricorsive. Per dire al compilatore che un case è ricorsivo basta apporre la keyword indirect:

enum ArithmeticExpression {
    case number ( Int )
    indirect case addition (ArithmeticExpression , ArithmeticExpression )
    indirect case multiplication (ArithmeticExpression , ArithmeticExpression )
}

Se tutti i case sono ricorsivi (tranne il case base ovvero il costruttore di base come ad esempio lo 0 per i numeri naturali) si può apporre indirect davanti alla keyword enum.

1
2
3
4
5
indirect enum ArithmeticExpression {
    case number ( Int )
    case addition (ArithmeticExpression , ArithmeticExpression )
    case multiplication (ArithmeticExpression , ArithmeticExpression )
}

Vediamo ora come si utilizza un tipo ricorsivo. Beh, è pur sempre un’enumerazione, soltanto che può ricevere come parametro un’enumeratore dello stesso tipo:

1
2
3
4
let five = ArithmeticExpression. number ( 5 )
let four = ArithmeticExpression. number ( 4 )
Balance New Donna gold black 500 Sneaker Nero
let sum = ArithmeticExpression. addition (five , four )
let product = ArithmeticExpression. multiplication (sum , ArithmeticExpression. number ( 2 ) )

L’induzione ci aiuta. Per delle nozioni sull’induzione vi rimandiamo qui. Adesso che abbiamo compreso come dichiarare ed istanziare un’enumerazione che rappresenta un tipo ricorsivo, vediamo come si utilizza:

1
2
3
4
5
6
7
8
9
10
11
12
func evaluate (_ expression : ArithmeticExpression ) -> Nero New black gold Donna Balance Sneaker 500 Int {
    switch expression {Con amp;n Donna Zeppa A Sandali Red 8q6wExT
        case let . number (value ) :
            return value
        case let . addition ( left , right ) Donna New black Nero Sneaker Balance 500 gold :
            return evaluate black New Donna Sneaker Nero gold Balance 500 ( left ) + evaluate ( right )
        case let . multiplication ( left , right ) :
            return evaluate ( left ) * evaluate ( right )
    }
}
print (evaluate (product ) )
// Stampa "18"

Semplicissimo, il comportamento è uguale a quello di una normalissima enumerazione ma ovviamente essendo le sue componenti dello stesso tipo, si può richiamare la funzione di elaborazione su queste e da qui il nome di tipo ricorsivo. Questa particolarità delle enum è interessante e risulta utile in vari contesti.

Abbiamo dunque esplicitato le operazioni sugli enum, dall’istanziazione alla creazione di funzioni che se ne servono anche per mezzo del matching con lo switch. Ricordiamo che l’enum è costante, immutabile, ma come vedremo poi esiste un modo per modificare internamente un enumeratore in Swift mediante la keyword mutating. Nella prossima lezione vedremo le Classi e le Strutture elementi chiave della logica OOP di Swift.

 

⇐ Chiusure                                                                        Mustang Donna Mustang Mustang Donna Blau Blau Blau Stivali Donna Stivali Mustang Stivali Stivali XwaFfOqx