Condividi tramite


Funzioni a valori scalari definite dall'utente - Scala

Questo articolo contiene esempi di funzioni definite dall'utente (UDF) Scala. Illustra come registrare funzioni definite dall'utente, come richiamare funzioni definite dall'utente e avvertenze relative all'ordine di valutazione delle sottoespressioni in Spark SQL. Per altri dettagli, vedere Funzioni scalari esterne definite dall'utente (UDF).

Nota

Le funzioni definite dall'utente scala nelle risorse di calcolo abilitate per catalogo Unity con modalità di accesso condiviso richiedono Databricks Runtime 14.2 e versioni successive.

Registrare una funzione come UDF

val squared = (s: Long) => {
  s * s
}
spark.udf.register("square", squared)

Richiamare l’UDF in Spark SQL

spark.range(1, 20).createOrReplaceTempView("test")
%sql select id, square(id) as id_squared from test

Usare la UDF con DataFrames

import org.apache.spark.sql.functions.{col, udf}
val squared = udf((s: Long) => s * s)
display(spark.range(1, 20).select(squared(col("id")) as "id_squared"))

Ordine di valutazione e controllo null

Spark SQL (inclusi SQL e le API di DataFrame e del set di dati) non garantisce l'ordine di valutazione delle sottoespressioni. In particolare, gli input di un operatore o di una funzione non vengono necessariamente valutati da sinistra a destra o in qualsiasi altro ordine fisso. Ad esempio, le espressioni logiche AND e OR non hanno semantica "corto circuito" da sinistra a destra.

Pertanto, è pericoloso basarsi sugli effetti collaterali o sull'ordine di valutazione delle espressioni booleane e sull'ordine delle clausole WHERE e HAVING, poiché tali espressioni e clausole possono essere riordinate durante l'ottimizzazione e la pianificazione delle query. In particolare, se una funzione definita dall'utente si basa sulla semantica di corto circuito in SQL per il controllo null, non esiste alcuna garanzia che il controllo null venga eseguito prima di richiamare la funzione definita dall'utente. Ad esempio,

spark.udf.register("strlen", (s: String) => s.length)
spark.sql("select s from test1 where s is not null and strlen(s) > 1") // no guarantee

Questa clausola WHERE non garantisce che la funzione definita dall'utente strlen venga richiamata dopo aver filtrato i valori null.

Per eseguire un controllo null appropriato, è consigliabile eseguire una delle operazioni seguenti:

  • Rendere la funzione definita dall'utente che riconosce i valori null ed eseguire il controllo null all'interno della funzione definita dall'utente stessa
  • Usare le espressioni IF o CASE WHEN per eseguire il controllo null e richiamare la funzione definita dall'utente in un ramo condizionale
spark.udf.register("strlen_nullsafe", (s: String) => if (s != null) s.length else -1)
spark.sql("select s from test1 where s is not null and strlen_nullsafe(s) > 1") // ok
spark.sql("select s from test1 where if(s is not null, strlen(s), null) > 1")   // ok

API del set di dati tipizzato

Nota

Questa funzionalità è supportata nei cluster abilitati per catalogo Unity con modalità di accesso condiviso in Databricks Runtime 15.4 e versioni successive.

Le API del set di dati tipizzato consentono di eseguire trasformazioni, ad esempio mapping, filtro e aggregazioni sui set di dati risultanti con una funzione definita dall'utente.

Ad esempio, l'applicazione Scala seguente usa l'API map() per modificare un numero in una colonna di risultato in una stringa con prefisso.

spark.range(3).map(f => s"row-$f").show()

Anche se questo esempio usa l'API map(), questo vale anche per altre API del set di dati tipizzato, ad esempio filter(), mapPartitions(), foreach(), foreachPartition(), reduce() e flatMap().