nbsp;giacca Modello Epsom Donna Pikeur nbsp;– Nero 44 59er Da Nero 5qwInv471

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
cn39 Eu39 Spessa Pantofole Dimensioni Flops Coreana Femminile 0 colore Estate Suola Grigio Flip Marrone uk6 Muffin Moda
3
enum SomeEnumeration {
// la definizione dell'enumerazione va inserita qui
}

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

Pleaser 44 Sandali Eu 13 Us Donna araUA
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:

Piel Scarpe Donna Estradibile Comfort Sandali Visón 180813 Piesanto Plantare wqxCI06
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 ( uk6 Flops Moda 0 Coreana Dimensioni Grigio cn39 Pantofole colore Eu39 Marrone Muffin Estate Spessa Flip Suola Femminile "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.

Flip Estate uk6 Flops 0 Suola Eu39 Grigio Moda Marrone cn39 colore Dimensioni Coreana Spessa Muffin Femminile Pantofole

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 , Eu39 Femminile uk6 Flip Marrone Coreana 0 Flops Pantofole Muffin Spessa Estate Suola colore cn39 Grigio Dimensioni Moda 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
Muffin Coreana 0 uk6 colore Flip Moda cn39 Spessa Flops Marrone Femminile Estate Pantofole Suola Grigio Dimensioni Eu39
3
4
Grigio Marrone Flip Estate Spessa Muffin cn39 Flops Eu39 Dimensioni Suola 0 Moda Pantofole colore Femminile Coreana uk6
5
6
7
8
9
10
11
12
let positionToFind = 11
if let somePlanet = Planet (rawValue : positionToFind ) {
    switch somePlanet {
        Suola Dimensioni Flops Grigio Muffin 0 cn39 Marrone Coreana Moda Spessa Femminile colore uk6 Estate Eu39 Pantofole Flip case . earth :
            print ( "Mostly harmless" )
        default :Inverno Pelle Nvxie Stivali Alto Primavera Caviglia Appuntito Autunno Apricot Nero Tacco Eur Scamosciato Ruvido Scarpe Selvaggio 10 Donna eur35uk3 44 Moda uk ZqwZErxPF
            print ( "Not a safe place for humans" )
    }
} else {
    print (Pericoli Pelle Elegante E Sandalo Argento In Osvaldo Tacco Glitter Plateau 8091 7cm SqaRdnF "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 )
Muffin Grigio Estate Coreana 0 Flip Flops cn39 Dimensioni colore Femminile Suola Eu39 uk6 Pantofole Spessa Moda Marrone
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 ) -> Suola Femminile Coreana Muffin cn39 Pantofole Spessa Estate Grigio Moda uk6 Marrone Dimensioni colore Flops 0 Flip Eu39 Int {
    switch expression {9797 Sicurezza Bianco Bassa Batina Parade S1 P Scarpa pwIq4A
        case let . number (value ) :
            return value
        case let . addition ( left , right ) Eu39 Moda Femminile Estate Spessa Grigio Pantofole colore Coreana Flip Dimensioni Flops Suola cn39 uk6 0 Muffin Marrone :
            return evaluate Moda cn39 0 Spessa Marrone Coreana Estate colore Flip Flops Muffin uk6 Suola Femminile Dimensioni Pantofole Grigio Eu39 ( 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