Scala - Estrattori

Un estrattore in Scala è un oggetto che ha un metodo chiamato unapplycome uno dei suoi membri. Lo scopo di quel metodo di non applicazione è quello di abbinare un valore e smontarlo. Spesso, l'oggetto estrattore definisce anche un doppio metodoapply per costruire valori, ma questo non è richiesto.

Esempio

Prendiamo un esempio di oggetto che definisce entrambi apply e unapplymetodi. Il metodo apply ha lo stesso significato di sempre: trasforma Test in un oggetto che può essere applicato agli argomenti tra parentesi nello stesso modo in cui viene applicato un metodo. Quindi puoi scrivere Test ("Zara", "gmail.com") per costruire la stringa "[email protected]".

Il unapply è ciò che trasforma la classe Test in un file extractor e inverte il processo di costruzione di apply. Dove apply prende due stringhe e forma da esse una stringa di indirizzo e-mail, unapply prende un indirizzo e-mail e restituisce potenzialmente due stringhe:user e il domain dell'indirizzo.

Il unapplydeve anche gestire il caso in cui la stringa data non è un indirizzo di posta elettronica. Ecco perché unapply restituisce un tipo di opzione su coppie di stringhe. Il suo risultato è uno dei dueSome (user, domain)se la stringa str è un indirizzo e-mail con le parti utente e dominio specificate, o Nessuno, se str non è un indirizzo e-mail. Ecco alcuni esempi come segue.

Sintassi

unapply("[email protected]") equals Some("Zara", "gmail.com")
unapply("Zara Ali") equals None

Il seguente programma di esempio mostra un oggetto estrattore per gli indirizzi di posta elettronica.

Esempio

object Demo {
   def main(args: Array[String]) {
      println ("Apply method : " + apply("Zara", "gmail.com"));
      println ("Unapply method : " + unapply("[email protected]"));
      println ("Unapply method : " + unapply("Zara Ali"));
   }
   
   // The injection method (optional)
   def apply(user: String, domain: String) = {
      user +"@"+ domain
   }

   // The extraction method (mandatory)
   def unapply(str: String): Option[(String, String)] = {
      val parts = str split "@"
      
      if (parts.length == 2){
         Some(parts(0), parts(1)) 
      } else {
         None
      }
   }
}

Salva il programma sopra in formato Demo.scala. I seguenti comandi vengono utilizzati per compilare ed eseguire questo programma.

Comando

\>scalac Demo.scala
\>scala Demo

Produzione

Apply method : [email protected]
Unapply method : Some((Zara,gmail.com))
Unapply method : None

Pattern Matching con estrattori

Quando un'istanza di una classe è seguita da parentesi con un elenco di zero o più parametri, il compilatore richiama il file applymetodo su quell'istanza. Possiamo definire applicare sia negli oggetti che nelle classi.

Come accennato in precedenza, lo scopo di unapplymetodo è estrarre un valore specifico che stiamo cercando. Fa l'operazione oppostaapplyfa. Quando si confronta un oggetto estrattore utilizzando l'estensionematch dichiarazione il unapply metodo verrà eseguito automaticamente.

Prova il seguente programma di esempio.

Esempio

object Demo {
   def main(args: Array[String]) {
      val x = Demo(5)
      println(x)

      x match {
         case Demo(num) => println(x+" is bigger two times than "+num)
         
         //unapply is invoked
         case _ => println("i cannot calculate")
      }
   }
   def apply(x: Int) = x*2
   def unapply(z: Int): Option[Int] = if (z%2==0) Some(z/2) else None
}

Salva il programma sopra in formato Demo.scala. I seguenti comandi vengono utilizzati per compilare ed eseguire questo programma.

Comando

\>scalac Demo.scala
\>scala Demo

Produzione

10
10 is bigger two times than 5