Förstå Apache Spark-kod för U-SQL-utvecklare
Viktigt!
Azure Data Lake Analytics drogs tillbaka den 29 februari 2024. Läs mer med det här meddelandet.
För dataanalys kan din organisation använda Azure Synapse Analytics eller Microsoft Fabric.
Det här avsnittet innehåller vägledning på hög nivå om hur du omvandlar U-SQL-skript till Apache Spark.
- Det börjar med en jämförelse av de två språkens bearbetningsparadigm
- Innehåller tips om hur du:
- Transformera skript inklusive U-SQL:s raderuppsättningsuttryck
- .NET-kod
- Datatyper
- Katalogobjekt
Förstå språk och bearbetningsparadigm för U-SQL och Spark
Innan du börjar migrera Azure Data Lake Analytics U-SQL-skript till Spark är det användbart att förstå de två systemens allmänna språk och bearbetningsfilosofi.
U-SQL är ett SQL-liknande deklarativt frågespråk som använder ett dataflödesparadigm och gör att du enkelt kan bädda in och skala ut användarkod skriven i .NET (till exempel C#), Python och R. Användartilläggen kan implementera enkla uttryck eller användardefinierade funktioner, men kan också ge användaren möjlighet att implementera så kallade användardefinierade operatorer som implementerar anpassade operatorer för att utföra transformeringar på raduppsättningsnivå, extrahering och skrivning av utdata.
Spark är ett skalbart ramverk som erbjuder flera språkbindningar i Scala, Java, Python, .NET osv. där du främst skriver koden på något av dessa språk, skapar dataabstraktioner som kallas elastiska distribuerade datamängder (RDD), dataramar och datauppsättningar och sedan använder ett LINQ-liknande domänspecifikt språk (DSL) för att transformera dem. Det ger också SparkSQL som en deklarativ underspråk på dataramen och datauppsättningsabstraktioner. DSL innehåller två kategorier av åtgärder, transformeringar och åtgärder. Om du tillämpar transformeringar på dataabstraktionerna körs inte transformeringen utan bygger i stället upp den körningsplan som ska skickas för utvärdering med en åtgärd (till exempel att skriva resultatet till en tillfällig tabell eller fil eller skriva ut resultatet).
När du översätter ett U-SQL-skript till ett Spark-program måste du därför bestämma vilket språk du vill använda för att åtminstone generera dataramsabstraktionen (som för närvarande är den mest använda dataabstraktionen) och om du vill skriva de deklarativa dataflödesomvandlingarna med hjälp av DSL eller SparkSQL. I vissa mer komplexa fall kan du behöva dela upp ditt U-SQL-skript i en sekvens med Spark och andra steg som implementeras med Azure Batch eller Azure Functions.
Dessutom erbjuder Azure Data Lake Analytics U-SQL i en serverlös jobbtjänstmiljö där resurser allokeras för varje jobb, medan Azure Synapse Spark, Azure Databricks och Azure HDInsight erbjuder Spark antingen i form av en klustertjänst eller med så kallade Spark-poolmallar. När du transformerar ditt program måste du ta hänsyn till konsekvenserna av att nu skapa, ändra storlek, skala och inaktivera kluster eller pooler.
Transformera U-SQL-skript
U-SQL-skript följer följande bearbetningsmönster:
- Data läse från antingen ostrukturerade filer, med instruktionen
EXTRACT
, en plats- eller filuppsättningsspecifikation och den inbyggda eller användardefinierade extraktorn och önskat schema, eller från U-SQL-tabeller (hanterade eller externa tabeller). Den representeras som en raduppsättning. - Raduppsättningarna transformeras i flera U-SQL-uttryck som tillämpar U-SQL-uttryck på raduppsättningarna och skapar nya rader.
- Slutligen matas de resulterande raduppsättningarna ut i antingen filer med hjälp av instruktionen
OUTPUT
som anger platser och en inbyggd eller användardefinierad utdata, eller till en U-SQL-tabell.
Skriptet utvärderas lazily, vilket innebär att varje extraherings- och transformeringssteg består av ett uttrycksträd och utvärderas globalt (dataflödet).
Spark-program liknar dem eftersom du skulle använda Spark-anslutningsappar för att läsa data och skapa dataramarna, tillämpa transformeringarna på dataramarna med hjälp av antingen LINQ-liknande DSL eller SparkSQL och sedan skriva resultatet till filer, tillfälliga Spark-tabeller, vissa programmeringsspråktyper eller konsolen.
Transformera .NET-kod
U-SQL:s uttrycksspråk är C# och erbjuder olika sätt att skala ut anpassad .NET-kod med användardefinierade funktioner, användardefinierade operatorer och användardefinierade aggregeringar.
Azure Synapse och Azure HDInsight Spark stöder nu internt körning av .NET-kod med .NET för Apache Spark. Det innebär att du kan återanvända vissa eller alla . NET-användardefinierade funktioner med Spark. Observera dock att U-SQL använder .NET Framework medan .NET för Apache Spark baseras på .NET Core 3.1 eller senare.
U-SQL-användardefinierade operatorer (U-SQL UDO) använder U-SQL UDO-modellen för att tillhandahålla utskalad körning av operatorns kod. Därför måste UDF:er skrivas om till användardefinierade funktioner för att passa in i Spark-körningsmodellen.
.NET för Apache Spark stöder för närvarande inte användardefinierade aggregatorer. Därför måste U-SQL-användardefinierade aggregatorer översättas till Spark-användardefinierade aggregeringar skrivna i Scala.
Om du inte vill dra nytta av .NET för Apache Spark-funktioner måste du skriva om dina uttryck till en motsvarande Spark, Scala, Java eller Python-uttryck, funktion, aggregator eller anslutningsapp.
Om du i vilket fall som helst har en stor mängd .NET-logik i dina U-SQL-skript kontaktar du oss via din Microsoft-kontorepresentant för ytterligare vägledning.
Följande information gäller för olika fall av .NET- och C#-användning i U-SQL-skript.
Transformera skalära infogade U-SQL C#-uttryck
U-SQL:s uttrycksspråk är C#. Många av de skalära infogade U-SQL-uttrycken implementeras internt för bättre prestanda, medan mer komplexa uttryck kan köras genom anrop till .NET-ramverket.
Spark har ett eget skalärt uttrycksspråk (antingen som en del av DSL eller SparkSQL) och tillåter anrop till användardefinierade funktioner som skrivits för JVM-, .NET- eller Python-körningen.
Om du har skalära uttryck i U-SQL bör du först hitta det lämpligaste inbyggda Spark-skaläruttrycket för att få mest prestanda och sedan mappa de andra uttrycken till en användardefinierad funktion av det Spark-körningsspråk som du väljer.
Tänk på att .NET och C# har olika typer av semantik än JVM- och Python-körningarna och Sparks DSL. Mer information om systemskillnaderna finns nedan .
Transformera användardefinierade skalära .NET-funktioner och användardefinierade aggregatorer
U-SQL tillhandahåller sätt att anropa godtyckliga skalära .NET-funktioner och anropa användardefinierade aggregatorer som skrivits i .NET.
Spark erbjuder också stöd för användardefinierade funktioner och användardefinierade aggregatorer som skrivits på de flesta av dess värdspråk som kan anropas från Sparks DSL och SparkSQL.
Som nämnts ovan stöder .NET för Apache Spark användardefinierade funktioner skrivna i .NET, men stöder inte användardefinierade aggregatorer. För användardefinierade funktioner kan .NET för Apache Spark användas, medan användardefinierade aggregatorer måste skrivas i Scala för Spark.
Transformera användardefinierade operatorer (UDF)
U-SQL innehåller flera kategorier av användardefinierade operatorer (UDF: er) som extraktorer, utdata, reducers, processorer, tillämpare och kombinationer som kan skrivas i .NET (och - till viss del - i Python och R).
Spark erbjuder inte samma utökningsmodell för operatörer, men har motsvarande funktioner för vissa.
Spark-motsvarigheten till extraktorer och utdata är Spark-anslutningsappar. För många U-SQL-extraktorer kan du hitta en motsvarande anslutningsapp i Spark-communityn. För andra måste du skriva en anpassad anslutningsapp. Om U-SQL-extraktorn är komplex och använder flera .NET-bibliotek kan det vara bättre att skapa en anslutningsapp i Scala som använder interop för att anropa till .NET-biblioteket som utför den faktiska bearbetningen av data. I så fall måste du distribuera .NET Core-körningen till Spark-klustret och se till att de refererade .NET-biblioteken är .NET Standard 2.0-kompatibla.
De andra typerna av U-SQL UDOs måste skrivas om med hjälp av användardefinierade funktioner och aggregeringar och det semantiskt lämpliga Spark DLS- eller SparkSQL-uttrycket. En processor kan till exempel mappas till en SELECT med olika UDF-anrop, paketerade som en funktion som tar en dataram som argument och returnerar en dataram.
Transformera U-SQL:s valfria bibliotek
U-SQL tillhandahåller en uppsättning valfria bibliotek och demobibliotek som erbjuder Python, R, JSON, XML, AVRO-stöd och vissa Funktioner för Azure AI-tjänster.
Spark erbjuder sin egen Python- och R-integrering, pySpark respektive SparkR, och tillhandahåller anslutningsappar för att läsa och skriva JSON, XML och AVRO.
Om du behöver transformera ett skript som refererar till Azure AI-tjänstbiblioteken rekommenderar vi att du kontaktar oss via din Microsoft-kontorepresentant.
Transformera inskrivna värden
Eftersom U-SQL:s typsystem baseras på .NET-typsystemet och Spark har ett eget typsystem som påverkas av värdspråkbindningen måste du se till att de typer du arbetar med är nära och för vissa typer kan typintervallen, precisionen och/eller skalan vara något annorlunda. Dessutom behandlar null
U-SQL och Spark värden på olika sätt.
Datatyper
Följande tabell innehåller motsvarande typer i Spark, Scala och PySpark för de angivna U-SQL-typerna.
U-SQL | Spark | Scala | PySpark |
---|---|---|---|
byte |
|||
sbyte |
ByteType |
Byte |
ByteType |
int |
IntegerType |
Int |
IntegerType |
uint |
|||
long |
LongType |
Long |
LongType |
ulong |
|||
float |
FloatType |
Float |
FloatType |
double |
DoubleType |
Double |
DoubleType |
decimal |
DecimalType |
java.math.BigDecimal |
DecimalType |
short |
ShortType |
Short |
ShortType |
ushort |
|||
char |
Char |
||
string |
StringType |
String |
StringType |
DateTime |
DateType , TimestampType |
java.sql.Date , java.sql.Timestamp |
DateType , TimestampType |
bool |
BooleanType |
Boolean |
BooleanType |
Guid |
|||
byte[] |
BinaryType |
Array[Byte] |
BinaryType |
SQL.MAP<K,V> |
MapType(keyType, valueType, valueContainsNull) |
scala.collection.Map |
MapType(keyType, valueType, valueContainsNull=True) |
SQL.ARRAY<T> |
ArrayType(elementType, containsNull) |
scala.collection.Seq |
ArrayType(elementType, containsNull=True) |
Mer information finns i:
Behandling av NULL
I Spark tillåter typer per standard NULL-värden medan du i U-SQL uttryckligen markerar skalärt, icke-objekt som nullbart. Spark gör att du kan definiera en kolumn som inte nullbar, men den framtvingar inte villkoret och kan leda till fel resultat.
I Spark anger NULL att värdet är okänt. Ett Spark NULL-värde skiljer sig från valfritt värde, inklusive sig självt. Jämförelser mellan två Spark NULL-värden, eller mellan ett NULL-värde och ett annat värde, returnerar okänt eftersom värdet för varje NULL är okänt.
Det här beteendet skiljer sig från U-SQL, som följer C#-semantiken där null
skiljer sig från alla värden men lika med sig själv.
Därför returnerar en SparkSQL-instruktion som använder WHERE column_name = NULL
noll rader även om det finns NULL-värden i column_name
, medan det i U-SQL returnerar raderna där column_name
är inställt på null
.SELECT
På samma sätt returnerar en Spark-instruktion SELECT
som använder WHERE column_name != NULL
noll rader även om det finns icke-null-värden i column_name
, medan den i U-SQL returnerar raderna som inte har null. Om du vill ha semantiken U-SQL null-check bör du därför använda isull respektive isnotnull (eller deras DSL-motsvarighet).
Transformera U-SQL-katalogobjekt
En stor skillnad är att U-SQL-skript kan använda sina katalogobjekt, varav många inte har någon direkt Spark-motsvarighet.
Spark har stöd för Hive Meta Store-begrepp, främst databaser, tabeller och vyer, så du kan mappa U-SQL-databaser och scheman till Hive-databaser och U-SQL-tabeller till Spark-tabeller (se Flytta data som lagras i U-SQL-tabeller), men det har inget stöd för tabellvärdesfunktioner (TVF:er), lagrade procedurer, U-SQL-sammansättningar, externa datakällor osv.
U-SQL-kodobjekt som vyer, TVF:er, lagrade procedurer och sammansättningar kan modelleras via kodfunktioner och bibliotek i Spark och refereras med hjälp av värdspråkets funktions- och procedurmässiga abstraktionsmekanismer (till exempel genom att importera Python-moduler eller referera till Scala-funktioner).
Om U-SQL-katalogen har använts för att dela data och kodobjekt mellan projekt och team måste motsvarande mekanismer för delning användas (till exempel Maven för att dela kodobjekt).
Transformera U-SQL-raduppsättningsuttryck och SQL-baserade skaläruttryck
U-SQL:s kärnspråk omvandlar raduppsättningar och baseras på SQL. Följande är en outtömmande lista över de vanligaste raduppsättningsuttrycken som erbjuds i U-SQL:
SELECT
/FROM
/WHERE
/GROUP BY
+Aggregeringar+HAVING
/ORDER BY
+FETCH
INNER
/OUTER
/CROSS
/SEMI
JOIN
UttryckCROSS
/OUTER
APPLY
UttryckPIVOT
/UNPIVOT
UttryckVALUES
raduppsättningskonstruktorAnge uttryck
UNION
/OUTER UNION
/INTERSECT
/EXCEPT
Dessutom tillhandahåller U-SQL olika SQL-baserade skaläruttryck, till exempel
OVER
fönsteruttryck- olika inbyggda sammansättningar och rangordningsfunktioner (
SUM
FIRST
osv.) - Några av de mest välbekanta SQL-skaläruttrycken:
CASE
,LIKE
, (NOT
)IN
,AND
osvOR
.
Spark erbjuder motsvarande uttryck i både dess DSL- och SparkSQL-formulär för de flesta av dessa uttryck. Vissa av de uttryck som inte stöds internt i Spark måste skrivas om med hjälp av en kombination av de inbyggda Spark-uttrycken och semantiskt likvärdiga mönster. Till exempel OUTER UNION
måste översättas till motsvarande kombination av prognoser och fackföreningar.
På grund av den olika hanteringen av NULL-värden matchar en U-SQL-koppling alltid en rad om båda kolumnerna som jämförs innehåller ett NULL-värde, medan en koppling i Spark inte matchar sådana kolumner om inte explicita null-kontroller läggs till.
Transformera andra U-SQL-begrepp
U-SQL erbjuder också olika andra funktioner och begrepp, till exempel federerade frågor mot SQL Server-databaser, parametrar, skalära och lambda-uttrycksvariabler, systemvariabler, OPTION
tips.
Federerade frågor mot SQL Server-databaser/externa tabeller
U-SQL tillhandahåller datakällor och externa tabeller samt direkta frågor mot Azure SQL Database. Spark erbjuder inte samma objektabstraktioner, men det tillhandahåller Spark-anslutningsprogram för Azure SQL Database som kan användas för att köra frågor mot SQL-databaser.
U-SQL-parametrar och variabler
Parametrar och användarvariabler har motsvarande begrepp i Spark och deras värdspråk.
I Scala kan du till exempel definiera en variabel med nyckelordet var
:
var x = 2 * 3;
println(x)
U-SQL:s systemvariabler (variabler som börjar med @@
) kan delas in i två kategorier:
- Uppsättningsbara systemvariabler som kan ställas in på specifika värden för att påverka skriptbeteendet
- Informationssystemvariabler som frågar efter information på system- och jobbnivå
De flesta av de uppsättningsbara systemvariablerna har ingen direkt motsvarighet i Spark. Vissa informationssystemvariabler kan modelleras genom att skicka informationen som argument under jobbkörningen, andra kan ha en motsvarande funktion i Sparks värdspråk.
U-SQL-tips
U-SQL erbjuder flera syntaktiska sätt att ge tips till frågeoptimeraren och körningsmotorn:
- Ange en U-SQL-systemvariabel
- en
OPTION
sats som är associerad med raduppsättningsuttrycket för att tillhandahålla ett data- eller plantips - ett kopplingstips i syntaxen för kopplingsuttrycket (till exempel
BROADCASTLEFT
)
Sparks kostnadsbaserade frågeoptimerare har sina egna funktioner för att ge tips och finjustera frågeprestandan. Se motsvarande dokumentation.
Nästa steg
- Förstå Spark-dataformat för U-SQL-utvecklare
- .NET för Apache Spark
- Uppgradera dina stordataanalyslösningar från Azure Data Lake Storage Gen1 till Azure Data Lake Storage Gen2
- Transformera data med Spark-aktivitet i Azure Data Factory
- Transformera data med Hadoop Hive-aktivitet i Azure Data Factory
- Vad är Apache Spark i Azure HDInsight