Parametrisk polymorfism i programmeringsspråk och typteori är en egenskap hos semantiken i ett typsystem som låter dig bearbeta värden av olika typer på ett identiskt sätt, det vill säga att fysiskt exekvera samma kod för data av olika typer [1] [2] .
Parametrisk polymorfism anses vara den "sanna" formen av polymorfism [3] , vilket gör språket mer uttrycksfullt och kraftigt ökande kodåteranvändning . Traditionellt är det motsats till ad-hoc-polymorfism [1] , som ger ett enda gränssnitt till potentiellt olika kod för olika typer som är tillåtna i ett givet sammanhang, potentiellt inkompatibla ( statiskt eller dynamiskt ). I ett antal kalkyler, såsom teorin om kvalificerade typer , behandlas ad-hoc polymorfism som ett specialfall av parametrisk polymorfism.
Parametrisk polymorfism ligger till grund för typsystemen av språk i ML- familjen ; sådana typsystem kallas polymorfa. I gemenskaper av språk med icke-polymorfa typsystem (ättlingar till Algol och BCPL [4] ), kallas parametriskt polymorfa funktioner och typer " generaliserade ".
Termen " parametrisk polymorfism " används traditionellt för att hänvisa till typsäker parametrisk polymorfism, även om otypade former av den också existerar (se parametrisk polymorfism i C och C++ ) [4] . Nyckelbegreppet för typsäker parametrisk polymorfism, förutom den polymorfa funktionen , är den polymorfa typen .
En polymorf typ ( eng. polymorphic type ), eller en polytype ( eng. polytype ) är en typ som parametriseras av en annan typ. En parameter i typlagret kallas en typvariabel (eller typvariabel) .
Formellt studeras typpolymorfism i den polymorfiskt typade lambda-kalkylen , kallad System F.
Till exempel kan en funktion som appendsammanfogar två listor till en byggas oberoende av typen av listelement. Låt typvariabeln a beskriva typen av listelement. Sedan kan funktionen appendskrivas som " forall a. [a] × [a] -> [a]" (här betyder konstruktionen [a]typen " en lista, vars element har en typa " - syntaxen som används i Haskell-språket ). I det här fallet sägs typen vara parametriserad av en variabel aför alla värden a. På varje plats för tillämpning av specifika argument löses appendvärdet och varje omnämnande av det i typsignaturen ersätts med ett värde som motsvarar tillämpningskontexten. I det här fallet kräver funktionstypsignaturen att elementtyperna för båda listorna och resultatet är identiska .a
Uppsättningen av giltiga värden för en typvariabel ges genom kvantifiering . De enklaste kvantifierarna är universella (som i exemplet med append) och existentiella (se nedan).
En kvalificerad typ är en polymorf typ, dessutom utrustad med en uppsättning predikat som reglerar intervallet av giltiga värden för en parameter av denna typ. Med andra ord låter typkvalificering dig kontrollera kvantifieringen på ett godtyckligt sätt. Teorin om kvalificerade typer byggdes av Mark P. Jones 1992 [ 5] . Den tillhandahåller en gemensam motivering för de mest exotiska typsystemen, inklusive typklasser , utvidgbara beteckningaroch subtyper , och tillåter att specifika begränsningar formuleras exakt för specifika polymorfa typer, vilket etablerar förhållandet mellan parametriska och ad-hoc polymorfism ( överbelastning ), och mellan explicit och implicit överbelastning. Föreningen av en typ med ett predikat idenna teori kallas bevis . En algebra som liknar lambdakalkylen formuleras för bevis , inklusive abstraktion av bevis, tillämpning av bevis, etc. Att korrelera en term av denna algebra med en explicit överbelastad funktion kallas bevisöversättning .
Motiverande exempel för utvecklingen av en generaliserad teori var Wadler-Blott-typklasserna och typningen av utvidgbara poster genom Harper-Pearce-predikat [5] [6] .
Parametriskt polymorfa typsystem klassificeras i grunden enligt rang och predikativ egenskap . Dessutom urskiljs explicit och implicit polymorfism [7] och ett antal andra egenskaper. Implicit polymorfism tillhandahålls genom typinferens , vilket avsevärt förbättrar användbarheten men har begränsad uttrycksförmåga. Många praktiska parametriskt polymorfa språk separerar faserna av ett externt implicit typat språk och ett internt explicit typat språk .
Den mest allmänna formen av polymorfism är " impredikativ polymorfism av högre rang ". De mest populära restriktionerna för denna form är polymorfismen i rang 1 som kallas " prenex " och den predikativa polymorfismen . Tillsammans bildar de " predikativ prenexpolymorfism " liknande den som implementerats i ML och tidigare versioner av Haskell .
När typsystem blir mer komplexa blir typsignaturer så komplexa att deras fullständiga eller nästan fullständiga härledning betraktas av många forskare som en kritisk egenskap, vars frånvaro kommer att göra språket olämpligt för praktik [8] [9] . Till exempel, för en traditionell kombinator, har den mapfullständiga typsignaturen (med hänsyn till generisk kvantifiering) under typsäker undantagsflödesspårning följande form [10] [8] (som ovan betyder en lista med element av typ ):[a]a
Rangen av polymorfism visarhäckningsdjupet för kvantifierare av typvariabler tillåtna inom ramen för systemet . " Polymorfism av rang k " (för k > 1 ) låter dig specificera typvariabler efter polymorfa typer av rangordning som inte är högre än ( k - 1) . " Polymorfism av högre rang " låter dig sätta kvantifierare av typvariabler till vänster om ett godtyckligt antal pilar i typer av högre ordning .
Joe Wells bevisade [ 11] att typinferens för ett Currytypat System F inte kan avgöras för rang över 2, så explicit typanteckning måste användas när man använder högre rang [12] .
Det finns partiella inferentiella typsystem som kräver att endast icke-deriverbara typvariabler ska annoteras [13] [14] [15] .
Prenex polymorfismPolymorfism av rang 1 kallas ofta prenex polymorfism (från ordet "prenex" - se prenex normalform ). I ett prenex polymorft system kan typvariabler inte instansieras av polymorfa typer. Denna begränsning gör distinktionen mellan monomorfa och polymorfa typer väsentlig, varför polymorfa typer i prenex-systemet ofta kallas för " typing schemes " ( engelska typschemas ) för att skilja dem från "vanliga" (monomorfa) typer (monotyper). Som en konsekvens kan alla typer skrivas i formen när alla typvariabelkvantifierare är placerade i den yttersta (prenex) positionen, vilket kallas prenex normalform . Enkelt uttryckt är polymorfa funktionsdefinitioner tillåtna, men polymorfa funktioner kan inte skickas som argument till andra funktioner [16] [17] - polymorfiskt definierade funktioner måste monotypinstansieras före användning.
En nära motsvarighet är den så kallade " Let-polymorphism " eller " ML - style polymorphism " av Damas-Milner. Tekniskt sett har Let-polymorphism i ML ytterligare syntaktiska begränsningar, såsom " värdebegränsningen " förknippad med typsäkerhetsproblem vid användning av referenser (som inte förekommer i rena språk som Haskell och Clean ) [18] [19] .
Predikativ (villkorlig) polymorfism kräver att en typvariabel instansieras med en monotyp (inte en polytyp).
Predikativa system inkluderar intuitionistisk typteori och Nuprl .
Impredikativ polymorfismImpredikativ (ovillkorlig) polymorfism låter dig instansiera en typvariabel med en godtycklig typ - både monomorf och polymorf, inklusive typen som definieras. (Att tillåta, inom en kalkyl, ett system att rekursivt inkludera sig självt i sig självt kallas impredikativitet . Potentiellt kan detta leda till paradoxer som Russell eller Cantor [20] , men detta händer inte i fallet med ett utarbetat system . [21] .)
Impredikativ polymorfism låter dig överföra polymorfa funktioner till andra funktioner som parametrar, returnera dem som ett resultat, lagra dem i datastrukturer, etc., varför det också kallas förstklassig polymorfism . Detta är den mest kraftfulla formen av polymorfism, men utgör å andra sidan ett allvarligt problem för optimering och gör typinferens olöslig .
Ett exempel på ett impredikativt system är System F och dess förlängningar (se lambdakub ) [22] .
Traditionellt, i ättlingar till ML , kan en funktion endast vara polymorf när den ses "från utsidan" (det vill säga den kan appliceras på argument av olika typer), men dess definition kan bara innehålla monomorf rekursion (det vill säga typupplösning är gjort innan samtalet). Att utvidga rekonstruktion av ML-typ till rekursiva typer ger inga allvarliga svårigheter. Å andra sidan skapar kombinationen av typrekonstruktion med rekursivt definierade termer ett svårt problem som kallas polymorf rekursion . Mycroft och Meertens föreslog en polymorf typningsregel som tillåter att rekursiva anrop till en rekursiv funktion från sin egen kropp instansieras med olika typer. I den här kalkylen, känd som Milner-Mycroft-kalkylen, är typinferens oavgörlig . [23]
Produkttyper (även kända som " poster ") fungerar som den formella grunden för objektorienterad och modulär programmering. Deras dubbla par består av summatyper (även känd som " varianter ") [24] [25] [19] :
Tillsammans är de ett sätt att uttrycka alla komplexa datastrukturer och vissa aspekter av beteendet hos program .
Record calculi är ett generaliserat namn för problemet och ett antal av dess lösningar angående flexibiliteten hos produkttyper i programmeringsspråk under villkoret typsäkerhet [26] [27] [28] . Termen sträcker sig ofta till summatyper, och gränserna för begreppet " posttyp " kan variera från kalkyl till kalkyl (liksom själva begreppet " typ "). Termerna " registrera polymorfism " (som, återigen, ofta inkluderar variant polymorfism ) [27] , " modulus calculus " [9] och " strukturell polymorfism " används också.
Typprodukter och summor ( poster och varianter [ en ] ger flexibilitet när det gäller att konstruera komplexa datastrukturer, men begränsningarna hos många system av verklig typ hindrar ofta att deras användning är riktigt flexibel. Dessa begränsningar uppstår vanligtvis på grund av problem med effektivitet, typinferens eller helt enkelt användbarhet. [29]
Det klassiska exemplet är Standard ML , vars typsystem medvetet begränsades precis tillräckligt för att kombinera enkel implementering med uttrycksfullhet och matematiskt bevisbar typsäkerhet .
Exempel på postdefinition :
> val r = { namn = "Foo" , använd = sant }; (* val r: {namn: sträng, använd: bool} = {namn = "Foo", använd = sant} *)Åtkomst till fältvärdet med dess namn utförs av en prefixkonstruktion av formen #field record:
> val r1 = #namn r ; (* val r1 : string = "Foo" *)Följande funktionsdefinition över posten är korrekt:
> roligt namn1 ( x : { namn : sträng , ålder : int }) = # namn xOch följande genererar ett kompilatorfel att typen inte är helt löst :
> roligt namn2 x = #namn x (* olöst typ i deklarationen: {namn : '1, ...} *)Monomorfismen hos poster gör dem oflexibla, men effektiva [30] : eftersom den faktiska minnesplatsen för varje postfält är känd i förväg (vid kompilering), kompileras med namn till den till direkt indexering, som vanligtvis beräknas i en maskin instruktion. Men när man utvecklar komplexa skalbara system är det önskvärt att kunna abstrahera fält från specifika poster - till exempel för att definiera en universell fältväljare:
kul välj r l = #l rMen att sammanställa polymorf åtkomst till fält som kan vara i olika ordning i olika register är ett svårt problem, både ur säkerhetskontroll av operationer på språknivå och ur prestandasynpunkt på maskinkodsnivå. En naiv lösning kan vara att dynamiskt slå upp ordboken vid varje samtal (och skriptspråk använder den), men detta är uppenbarligen extremt ineffektivt. [31]
Typsummor utgör grunden för grenuttrycket , och på grund av typsystemets stränghet ger kompilatorn kontroll över analysens fullständighet. Till exempel för följande summatyp :
datatyp 'a foo = A för 'a | B av ( 'a * 'a ) | Cvilken funktion som helst över det kommer att se ut
fun bar ( p : 'a foo ) = fallet p av A x => ... | B ( x , y ) => ... | c => ...och när någon av satserna tas bort, kommer kompilatorn att utfärda en varning för att analysera ofullständig (" match nonexhaustive"). För fall där endast ett fåtal av de många alternativen kräver analys i ett givet sammanhang, är det möjligt att organisera default-förgrening med hjälp av den sk. "Joker" - ett universellt prov, med vilket alla giltiga (enligt skrivning) värden är jämförbara. Det är skrivet med ett understreck (" _"). Till exempel:
fun bar ( p : 'a foo ) = fallet p av C => ... | _ => ...I vissa språk (i Standard ML , i Haskell ) är till och med den "enklare" konstruktionen if-then-elsebara syntaktisk socker framför ett specialfall av förgrening , dvs uttrycket
om A så B annars Credan på stadium av grammatisk analys presenteras i formuläret
fallet A av sant => B | falskt => Celler
fallet A av sant => B | _ => CI Standard ML är poster och varianter monomorfa, men syntaktisk socker för att hantera poster med flera fält stöds, kallat det " universella mönstret " [32] :
> val r = { namn = "Foo" , använd = sant }; (* val r : {name : string, used : bool} = {name = "Foo", used = true} *) > val { used = u , ...} = r ; (* val u : bool = sant *)En ellips av {used, ...}typen " " betyder att det finns andra fält i denna post, utöver de som nämns. Det finns dock ingen postpolymorfism som sådan (inte ens prenexet ): full statisk upplösning av informationen om monomorfa posttyp (via slutledning eller explicit anteckning ) krävs.
Termen utvidgbara poster används för en generaliserad beteckning av sådana operationer som expansion (bygga en ny post baserad på en befintlig med tillägg av nya fält), skärning (ta en specificerad del från en befintlig post), etc. I synnerhet, det innebär möjligheten för den så kallade " funktionella postuppdateringen " ( funktionell postuppdatering ) - operationen att konstruera ett nytt postvärde baserat på det befintliga genom att kopiera namnen och typerna av dess fält, i vilka ett eller flera fält i det nya rekordet får nya värden som skiljer sig från de ursprungliga (och möjligen har en annan typ). [33] [34] [19] [35] [36] [37]
Funktionella uppdaterings- och förlängningsoperationer är i sig själva ortogonala för att registrera polymorfism, men deras polymorfa användning är av särskilt intresse. Även för monomorfa poster blir det viktigt att kunna utelämna explicit omnämnande av fält som kopieras oförändrade, och detta representerar faktiskt postpolymorfism i den rent syntaktiska formen . Å andra sidan, om vi betraktar alla poster i systemet som utökningsbara, så gör detta att funktioner på poster kan skrivas som polymorfa.
Ett exempel på den enklaste designvarianten är implementerad i Alice ML (enligt de nuvarande efterföljande ML- konventionerna ) [36] . Det universella mönstret (ellipsis ) har utökade möjligheter: det kan användas för att "fånga en rad" för att arbeta med den "återstående" delen av posten som ett värde:
> val r = { a = 1 , b = sant , c = "hej" } (* r = {a = 1, b = sant, c = "hej"} *) > val { a = n , ... = r1 } = r (* r1 = {b=true, c="hej"} *) > val r2 = { d = 3,14 , ... = r1 } (* r2 = {b=true, c="hej ", d=3,14} *)Funktionsuppdateringen implementeras som en derivata av att fånga en rad med ett serviceord where. Till exempel följande kod:
> låt val r = { a = 1 , c = 3,0 , d = "inte en lista" , f = [ 1 ], p = [ "inte en sträng" ], z = INGEN } i { r där d = noll , p = "hej" } slutkommer automatiskt att skrivas om i formen:
> låt val r = { a = 1 , c = 3,0 , d = "inte en lista" , f = [ 1 ], p = [ "inte en sträng" ], z = INGEN } val { d = _, p = _, ... = tmp } = r in { ... = tmp , d = noll , p = "hej" } slutEn av de första (slutet av 1980 -talet - början av 1990 -talet ) föreslog olika alternativ för att formalisera expanderbara poster genom sammanlänkningsoperationer på icke-polymorfa poster (Harper-Pearce [38] , Wand [39] , Salzmann [40] ). Cardelli använde till och med skivoperationer istället för polymorfism på bärnstensspråket. Det finns inget känt sätt att sammanställa dessa kalkyler effektivt; dessutom är dessa beräkningar mycket komplexa ur typteoretisk analyssynpunkt. [27] [41] [42] [43]
Till exempel [33] :
val bil = { namn = "Toyota" ; ålder = "gammal" ; id = 6678 } val lastbil = { namn = "Toyota" ; id = 19823235 } ... val repaired_truck = { bil och lastbil }radpolymorfismen ) visade att multipelt arv kan modelleras genom rekordsammansättning [39] [33] .
Luca Cardelli undersökte möjligheten att formalisera " postpolymorfism " genom att subtypa relationer på poster: för att göra detta behandlas en post som en "universell subtyp", det vill säga dess värde tillåts referera till hela uppsättningen av dess supertyper. Detta tillvägagångssätt stöder också metodarv och fungerar som en typteoretisk grund för de vanligaste formerna av objektorienterad programmering . [27] [44] [45]
Cardelli presenterade en variant på metoden att sammanställa rekordpolymorfism över deras undertyper genom att fördefiniera offseten av alla möjliga etiketter i en potentiellt enorm struktur med många tomma platser [31] .
Metoden har betydande nackdelar. Stöd för subtypning i typsystemet komplicerar i hög grad mekanismen för kontroll av typkonsistens [46] . Dessutom, i dess närvaro, upphör den statiska typen av uttrycket att ge fullständig information om den dynamiska strukturen för värdet på posten . Till exempel, när du endast använder undertyper, följande term:
> om sant så är { A = 1 , B = sant } annars { B = falskt , C = "Cat" } (* val it : {B : bool} *)har typ {B : bool}, men dess dynamiska värde är lika med {A = 1, B = true}, det vill säga information om typen av den utökade posten går förlorad [43] , vilket är ett allvarligt problem för att kontrollera operationer som kräver fullständig information om värdestrukturen för att de ska kunna köras (t.ex. jämförelse för jämställdhet ) [19] . Slutligen, i närvaro av undertyper, påverkar valet mellan ordnad och oordnad representation av poster allvarligt prestandan [47] .
Populariteten för subtyping beror på att det ger enkla och visuella lösningar på många problem. Till exempel kan objekt av olika typer placeras i en enda lista om de har en gemensam supertyp [48] .
Mitchell Wand föreslog 1987 idén att(inte uttryckligen specificerad) delen av posten genom vad han kallade en radtypsvariabel ( radtypsvariabel ) [49] .
En radvariabel är en variabel av typen som går genom en uppsättning ändliga uppsättningar (rader) av typade fält (par av " (значение : тип)"). Resultatet är förmågan att implementera breddövergripande arv direkt ovanpå den parametriska polymorfismen som ligger till grund för ML utan att komplicera typsystemet med undertypningsregler Den resulterande typen av polymorfism kallas radpolymorfism . Inline polymorfism sträcker sig till både produkter av typer och summor av typer .
Vand lånade termen från engelsmännen. rad (rad, kedja, linje) från Algol-68 , där det betydde en uppsättning vyer. I ryskspråkig litteratur har denna term i samband med Algol traditionellt översatts som "flerarter". Det finns också en översättning av "radvariabler" som "strängvariabler" [41] , vilket kan orsaka förvirring med små bokstäver i strängtyper .
Exempel ( OCaml- språk ; postfix-syntax, record#field) [50] :
# låt skicka_m a = a # m ;; (* värde send_m : < m : a; .. > -> a = <roligt> *)Här är ellipsen (av två punkter) OCamls accepterade syntax för en icke namngiven inline-typvariabel . På grund av sådan typning kan funktionen send_mappliceras på ett objekt av vilken som helst (ej tidigare känd ) objekttyp, vilket inkluderar en metod för mmotsvarande signatur.
Typavdraget för Wanda-kalkylen i originalversionen var ofullständigt: på grund av bristen på begränsningar för utvidgningen av serien kommer att lägga till ett fält om namnet matchar det befintliga - som ett resultat har inte alla program en huvudtyp [6] [43] . Detta system var dock det första konkreta förslaget att utöka ML med register som stöder arv [51] . Under de följande åren föreslogs ett antal olika förbättringar, inklusive de som gör det komplett [51] [27] .
Det mest anmärkningsvärda spåret lämnades av Didier Remy ( franska: Didier Rémy ). Han byggde ett praktiskt typsystem med utbyggbara register, inklusive en komplett och effektiv typrekonstruktionsalgoritm [52] [53] . Remy stratifierar språket av typer i sorter , formulerar en sorterad algebra av typer ( eng. sorterad algebra av typer, sorterad språk av typer ). En skillnad görs mellan den egentliga typen av typer (inklusive fälttyper) och typen av fält ; mappningar mellan dem införs och på grundval av dem formuleras reglerna för att skriva utökade poster som en enkel förlängning av de klassiska ML- reglerna . Närvaroinformationen för ett fält definieras som en mappning från en typsortering till en fältsortering . Variabler av radtyp omformuleras till variabler som hör till fältsorteringen och är lika med frånvarokonstanten , som är ett element i fältsorteringen som inte har en matchning i typsorteringen . En typutvärderingsoperation för en post med n fält definieras som att mappa ett n-ärt fält till en typ (där varje fält i tupeln antingen beräknas av närvarofunktionen eller ges av frånvarokonstanten ).
På ett förenklat sätt kan idén med kalkyl tolkas som en förlängning av typen av vilket fält som helst i posten med en närvaro/frånvaro-flagga och representation av posten som en tuppel med en lucka för varje möjligt fält [6] . I implementeringsprototypen gjordes syntaxen för typspråket närmare den typteoretiska formuleringen, till exempel [52] :
# låt bilen = { namn = "Toyota" ; ålder = "gammal" ; id = 7866 } ;; (* bil : ∏ (namn : pre (sträng); id : pre (num); ålder : pre (sträng); abs) *) # let truck = { name = "Blazer" ; id = 6587867567 } ;; (* lastbil : ∏ (namn : pre (sträng); id : pre (num); abs) *) # låt person = { namn = "Tim" ; ålder = 31 ; id = 5656787 } ;; (* person : ∏ (namn : pre (sträng); id : pre (num); ålder : pre (num); abs) *)(symbolen ∏i Remy betyder typberäkningsoperationen)
Att lägga till ett nytt fält skrivs med hjälp av konstruktionen with:
# låt förare = { person med fordon = bil } ;; (* förare : ∏ (fordon : pre (∏ (namn : pre (sträng); id : pre (num); ålder : pre (sträng); abs)); namn : pre (sträng); id : pre (num) ; ålder: pre (num); abs) *)Funktionell uppdatering är skriven identiskt, med skillnaden att omnämnandet av ett redan existerande fält åsidosätter det:
# let truck_driver = { chaufför med fordon = lastbil };; (* lastbilschaufför : ∏ (fordon : pre (∏ (namn : pre (sträng); id : pre (num); abs)); namn : pre (sträng); id : pre (num); ålder : pre (num ); magmuskler) *)Detta schema formaliserar den begränsning som behövs för att kontrollera operationer på poster och sluta sig till huvudtypen, men leder inte till en uppenbar och effektiv implementering [6] [43] . Remy använder hash, vilket är ganska effektivt i genomsnitt, men ökar runtime overhead även för inbyggt monomorfa program, och är dåligt lämpad för poster med många fält [31] .
Rémy fortsatte med att utforska användningen av inline polymorfism i samband med datasubtyping , och betonade att dessa är ortogonala begrepp och visar att poster blir mest uttrycksfulla när de används tillsammans [54] . På grundval av detta, tillsammans med Jérôme Vouillon , föreslog han en lättviktig objektorienterad förlängning till ML [55] . Denna förlängning implementerades i Xavier Leroys språk "Caml Special Light" , vilket gjorde det till OCaml [56] . OCaml- objektmodellen är nära sammanflätad med användningen av strukturell subtypning och inline polymorfism [48] , vilket är anledningen till att de ibland felaktigt identifieras. Inline produktpolymorfism i OCaml är kärnan i typinferensen ; subtypningsrelationer är inte nödvändiga i ett språk som stöds, men ökar flexibiliteten ytterligare i praktiken [55] [50] [48] . OKaml har en enklare och mer beskrivande syntax för typinformation.
Jacques Garrigue ( fr. Jacques Garrigue ) implementerade [25] ett praktiskt system av polymorfa summor . Han kombinerade Remi och Ohoris teoretiska arbete och byggde ett system som körs i mitten: information om förekomsten av taggar i en post representeras med hjälp av kön och information om deras typer använder inline-variabler. För att kompilatorn ska kunna skilja mellan polymorfa och monomorfa summor använder Garriga en speciell syntax (backtick, prefixtagg). Detta eliminerar behovet av en typdeklaration - du kan omedelbart skriva funktioner på den, och kompilatorn kommer att mata ut en minimal lista med taggar när dessa funktioner är sammansatta. Detta system blev en del av OCaml runt 2000 , men inte istället för , utan förutom monomorfa summor , eftersom de är något mindre effektiva, och på grund av oförmågan att kontrollera fullständigheten av analysen , gör det svårt att hitta fel (till skillnad från Blooms lösning ). [25] [57]
Nackdelarna med Wands inline-polymorfism är att implementeringen inte är självklar (det finns inget enda systematiskt sätt att kompilera den, varje specifik typsystem baserat på inline-variabler har sin egen implementering) och det tvetydiga förhållandet med teorin (det finns ingen enhetlig formulering för typkontroll och slutledning , stöd för slutledning löstes separat och krävde experiment med att införa olika restriktioner) [27] .
Den mest komplexa typen av poster är beroende poster. Sådana poster kan inkludera typer såväl som "vanliga" värden (materialiserade typer, reified [9] ), och termerna och typerna i nästa ordning i postens brödtext kan bestämmas baserat på de som föregår dem . Sådana beteckningar motsvarar de " svaga summorna " av beroendetypsteorin , även känd som " existentialer ", och fungerar som den mest allmänna motiveringen för modulsystem i programmeringsspråk [58] [59] . Cardelli ansåg [60] typer liknande egenskaper som en av huvudtyperna i fulltypsprogrammering (men kallade dem " tupler ").
Robert Harper och Mark Lillibridge konstruerade [61 ] [59] den genomskinliga summakalkylen för attformellt rättfärdiga ett förstklassigt modulspråk av högre ordning , det mest avancerade modulsystemet som är känt. Denna kalkyl används bland annat i Harper-Stone semantiken , som representerar den typteoretiska motiveringen för Standard ML .
Genomskinliga summor generaliserar svaga summor genom etiketter och en uppsättning likheter som beskriver typkonstruktörer . Termen genomskinlig betyder att en posttyp kan innehålla både typer med en explicit exporterad struktur, såväl som helt abstrakta . Lagret av släkten i kalkyl har en enkel klassisk sammansättning: kön av alla typer och funktionella släkten av slaget urskiljs , typtypkonstruktörer ( ML stöder inte polymorfism i högre släkten , alla typvariabler tillhör släktet , och abstraktionen av typkonstruktorer är endast möjlig genom funktorer [62] ). Kalkylen skiljer mellan subtypningsregler för poster som bastyper och för postfält som deras beståndsdelar, med hänsyn till "subtyper" och "subfields", medan döljande (abstraherande) fältsignaturer är ett separat koncept från subtyping. Subtyping här formaliserar kartläggningen av moduler till gränssnitt . [61] [9]
Harper-Lilybridge-kalkylen är obestämbar även när det gäller typkonsistenskontroll ( modulspråksdialekter implementerade i Standard ML och OCaml använder begränsade delmängder av denna kalkyl). Senare byggde dock Andreas Rossberg , baserat på deras idéer, språket " 1ML ", där de traditionella kärnnivåposterna för språk- och modulnivåstrukturerna är samma förstklassiga konstruktion [9] (betydligt mer uttrycksfulla än Cardelli's - se kritik av ML-modulens språk ). Genom att införliva idén från Harper och Mitchell [63] om att dela in alla typer i universum av "små" och "stora" typer (förenklat liknar detta den grundläggande uppdelningen av typer i enkla och aggregerade typer, med ojämlika typningsregler), tillhandahöll Rossberg lösbarhet inte bara konsistenskontroller , utan nästan fullständig typslutning . Dessutom tillåter 1ML impredikativ polymorfism [64] . Samtidigt är det interna språket i 1ML baserat på det platta System F ω och kräver inte användning av beroende typer som metateori. Från och med 2015 lämnade Rossberg möjligheten att lägga till inline polymorfism öppen till 1ML , och noterade bara att detta borde ge mer fullständig typslutning [9] . Ett år senare lade han [65] effekterna polymorfism till 1ML .
Atsushi Ohori , tillsammans med sin handledare Peter Buneman , föreslog 1988 idén att begränsa utbudet av möjliga värden för vanliga typvariabler i den polymorfa typningen av själva poster . Senare formaliserade Ohori denna idé genom notationssläktena , efter att 1995 ha byggt en fullfjädrad kalkyl och en metod för dess effektiva sammanställning [19] [30] . Implementeringsprototypen skapades 1992 som en förlängning av SML/NJ [66] kompilatorn, sedan ledde Ohori utvecklingen av sin egen SML# kompilator, som implementerar Standard ML dialekten med samma namn . I SML# fungerar postpolymorfism som grunden för att sömlöst bädda in SQL- konstruktioner i SML- program . SML# används av stora japanska företag för att lösa affärsproblem i samband med högbelastade databaser [67] . Ett exempel på en sådan session ( REPL ) [68] :
rolig rik { Lön = s , ... } = s > 100000 ; (* val rik = fn : 'a#{ Lön:int, ... } -> bool *) kul ung x = #Ålder x < 24 ; (* val young = fn : 'a#{ Age:int, ... } -> bool *) fun youngAndWealthy x = rik x och även ung x ; (* val youngAndWealthy = fn : 'a#{ Age:int, Lön:int, ... } -> bool *) roligt välj display l pred = fold ( fn ( x , y ) => if pred x then ( visa x ) ::y else y ) l noll ; (* val select = fn : ('a -> 'b) -> 'en lista -> ('a -> bool) -> 'b lista *) fun youngAndWealthyEmployees l = välj #Name l youngAndWealthy ; (* val youngAndWealthyEmployees = fn : 'b#{ Ålder:int, Namn:'a, Lön:int, ... } lista -> 'en lista *)Ohori kallade sin kalkyl för " record polymorphism " ( engelsk rekordpolymorfism ) eller " polymorphic record calculus " ( engelsk polymorphic record calculus ), samtidigt som han betonade att han, liksom Wand, betraktar polymorfism inte bara av produkttyper , utan också av typer- summor [27] .
Ohori-kalkylen kännetecknas av den mest intensiva användningen av släktskiktet [6] . I posten (referenstyp till genus ), betyder symbolen antingen släktet av alla typer ; eller postsläktet , som har formen , som betecknar uppsättningen av alla poster som innehåller åtminstone de angivna fälten; eller ett variantsläkte med formen som anger uppsättningen av alla varianttyper som innehåller åtminstone de specificerade konstruktörerna . I språkets platta syntax skrivs en typbegränsning till någon sorts notation som (se exempel ovan). Lösningen liknar något som den begränsade kvantifieringen Cardelli-Wegner [27] . t#{...}
Den enda polymorfa operationen som tillhandahålls av denna kalkyl är fältextraktionsoperationen [69] . Men Ohori var den första som introducerade ett enkelt och effektivt kompileringsschema för rekordpolymorfism [43] . Han kallade det "realisationskalkylen" ( implementeringskalkyl ). En post representeras av en vektor ordnad lexikografiskt efter fältnamnen på den ursprungliga posten; adressering av ett fält med namn översätts till ett anrop till en mellanfunktion som returnerar numret på det givna fältet i den givna vektorn med det begärda namnet, och efterföljande indexering av vektorn med det beräknade positionsnumret. Funktionen anropas endast när man instansierar polymorfa termer, vilket medför en minimal overhead på körtid när man använder polymorfism och inte lägger någon overhead när man arbetar med monomorfa typer. Metoden fungerar lika bra med godtyckligt stora poster och varianter. Kalkylen ger typinferens och finner en stark överensstämmelse med teorin ( generisk kvantifiering är direkt relaterad till den vanliga vektorindexeringen ), som är en direkt förlängning av Girard-Reynolds andra ordningens lambdakalkyl , som tillåter olika välkända egenskaper hos polymorfa typning som ska överföras till rekordpolymorfism [31] .
I praktiken har stöd för polymorfa varianter i SML# inte implementerats på grund av dess inkompatibilitet med Standard ML :s definitionsmekanism för summatyp (kräver syntaktisk separation av summor och rekursiva typer) [70] .
Nackdelen med Ohori-kalkylen är bristen på stöd för rekordexpansion eller trunkeringsoperationer [27] [71] [43] .
I teorin om kvalificerade typer beskrivs expanderbara poster av frånvaron av ett fält ( "saknar" predikat ) och närvaron av ett fält ( "har" predikat ) predikat. Benedict Gaster ( Benedict R. Gaster ) tillsammans med författaren till teorin Mark Jones ( Mark P. Jones ) slutförde den i termer av utvidgbara poster till ett praktiskt typsystem av implicit maskinskrivna språk, inklusive definition av kompileringsmetoden [6] . De myntar termen förstklassiga etiketter för att betona förmågan att abstrahera fältoperationer från statiskt kända etiketter. Senare försvarade Gaster sin avhandling [72] om det konstruerade systemet .
Gaster-Jones-kalkylen ger inte full typ av slutledning . På grund av beslutbarhetsproblem infördes dessutom en konstgjord begränsning: ett förbud mot tomma serier [73] . Sulzmann försökte omformulera kalkylen [40] , men systemet han byggde kan inte utökas för att stödja polymorf rekordexpansion och har inte en universell effektiv kompileringsmetod [43] .
Daan Leijen lade till ett radlikhetspredikat (eller radlikhetspredikat ) till Gaster-Jones-kalkylen och flyttade seriespråket till predikatspråket - detta säkerställde fullständig typslutning och upphävde förbudet mot tomma serier [74] . Vid sammanställning omvandlas protokollen till en lexikografiskt ordnad tupel och bevisöversättning tillämpas enligt Gaster-Jones-schemat. Layens system tillåter uttryck av idiom som meddelanden av högre ordning (den mest kraftfulla formen av objektorienterad programmering ) och förstklassiga grenar .
Baserat på dessa arbeten har tillägg till Haskell-språket [75] implementerats .
Resultaten av Gaster-Jones är mycket nära Ohoris , trots betydande skillnader i typteoretisk motivering, och den största fördelen är stödet för rekordexpansion och trunkeringsoperationer [6] . Nackdelen med kalkyl är att den förlitar sig på egenskaper hos typsystemet som inte finns i de flesta programmeringsspråk. Dessutom är typinferens för det ett allvarligt problem, varför författarna har infört ytterligare begränsningar. Och även om Layen har eliminerat många av bristerna, är det inte möjligt att använda -grenen [71]default
Med utvecklingen av mjukvarusystem kan antalet alternativ i summatypen öka och att lägga till varje alternativ kräver att man lägger till en motsvarande gren till varje funktion över denna typ, även om dessa grenar är identiska i olika funktioner. Således beror komplexiteten i att öka funktionaliteten i de flesta programmeringsspråk icke-linjärt på deklarativa ändringar i referensvillkoren. Detta mönster är känt som uttrycksproblemet . Ett annat välkänt problem är undantagshantering : under decennier av forskning om typsystem kunde alla språk som klassificerats som typsäkra fortfarande lämnas genom att kasta ett oupptäckt undantag - eftersom, trots att själva undantagen skrivits, mekanismen för att höja och hanteringen av dem var inte maskinskriven. Även om verktyg för att analysera flödet av undantag har byggts, har dessa verktyg alltid varit externa i förhållande till språket.
Matthias Blume , en kollega till Andrew Appel som arbetar med projektets efterföljare ML [76] , hans doktorand Wonseok Chae och kollega Umut Acar löste både problem baserade på matematisk dualitet produkter och summor . Förkroppsligandet av deras idéer var språket MLPolyR [71] [34] [77] , baserat på den enklaste delmängden av Standard ML och kompletterande den med flera nivåer av typsäkerhet [78] . Wonseok Chai försvarade senare sin avhandling om dessa prestationer [78] .
Lösningen är följande. Enligt principen om dualitet motsvarar introduktionsformen för ett begrepp elimineringsformen för dess dual [ 71] . Således motsvarar elimineringsformen för summor (analys av grenar) den inledande formen för poster. Detta uppmuntrar förgrening att ha samma egenskaper som redan är tillgängliga för poster – gör dem till förstklassiga objekt och låt dem utökas.
Till exempel, det enklaste uttrycket språktolk:
fun eval e = fall e av `Const i => i | `Plus (e1,e2) => eval e1 + eval e2med införandet av förstklassig konstruktion caseskan skrivas om som:
fun eval e = matcha e med fall `Const i => i | `Plus (e1,e2) => eval e1 + eval e2varefter cases-blocket kan renderas:
fun eval_c eval = fall `Const i => i | `Plus (e1,e2) => eval e1 + eval e2 fun eval e = matcha e med (eval_c eval)Denna lösning möjliggör default-förgrening (till skillnad från Gaster-Jones kalkylen ), vilket är viktigt för sammansättningen av förstklassiga grenar [34] . Komplettering av radens sammansättning utförs med hjälp av ordet . nocases
fun const_c d = fall `Const i => i standard : d fun plus_c eval d = fall `Plus (e1,e2) => eval e1 + eval e2 default : d fun eval e = matcha e med const_c (plus_c eval nocases ) fun bind env v1 x v2 = om v1 = v2 så x else env v2 fun var_c env d = fall `Var v => env v default : d fun let_c eval env d = fall `Let (v,e,b) => eval (bind env v (eval env e)) b default : d fun eval_var env e = matcha e med const_c (plus_c (eval_var env) (var_c env (let_c eval_var env nocases )))Som du kan se kräver den nya koden, som måste läggas till med den kvalitativa komplikationen av systemet, inte ändra den redan skrivna koden (funktionerna const_coch plus_c"vet ingenting" om det efterföljande tillägget av stöd för variabler och let-block till språktolken). Således är förstklassiga utvidgbara fall en grundläggande lösning på uttrycksproblemet , vilket gör att man kan tala om ett utvidgbart programmeringsparadigm [71] [78] . inline polymorphism , som redan stöds i typsystemet, och i denna mening är fördelen med en sådan teknisk lösning dess konceptuella enkelhet [ 34] .
Utvidgning av mjukvarusystem kräver dock också kontroll över hanteringen av undantag som kan kastas på godtyckliga anropskapslingsdjup. Och här proklamerar Bloom och kollegor en ny typsäkerhetsslogan i utvecklingen av Milners slogan : " Program som klarar typkontroll kastar inte obehandlade undantag ." Problemet är att om funktionstypens signatur innehåller information om de typer av undantag som denna funktion potentiellt kan ge, och denna information i signaturen för den godkända funktionen måste vara strikt överensstämmande med informationen om funktionsparametern av högre ordning (inklusive om detta är en tom uppsättning ) - att skriva undantagshanteringsmekanismen kräver omedelbart polymorfismen hos själva undantagstyperna - annars upphör funktioner av högre ordning att vara polymorfa. Samtidigt, i ett säkert språk, är undantagen en utvidgbar summa [79] [80] [81] , det vill säga en varianttyp vars konstruktorer läggs till allt eftersom programmet fortskrider. Följaktligen innebär säkerhet av undantagsflödestyp behovet av att stödja summatyper som är både töjbara och polymorfa. Även här är lösningen inline polymorfism .
Liksom Garriga-kalkylen , använder MLPolyR en speciell syntax för polymorfa summor (backtick, ledande tagg), och det finns inget behov av en fördeklaration av summatypen (det vill säga ovanstående kod är hela programmet, inte ett fragment). Fördelen är att det inte finns några problem med att analysera fullständighetskontroll: semantiken för MLPolyR definieras genom att konvertera till ett bevisat tillförlitlighet internt språk som varken stöder summapolymorfism eller undantag (för att inte tala om oförfångade undantag), så behovet av dem radering vid kompilering är i sig ett bevis på tillförlitlighet. [34]
MLPolyR använder en icke-trivial kombination av flera kalkyler och tvåstegsöversättning. Den använder Remy-kalkylen för typavdrag och typmatchning , samtidigt som den använder principen om dualitet för att representera summor som produkter, sedan översätter språket till ett mellanliggande uttryckligt maskinskrivet språk med polymorfa poster, och använder sedan Ohoris effektiva kompileringsmetod . Med andra ord, Ohoris kompileringsmodell har generaliserats för att stödja Remy-kalkylen [69] . På typteoretisk nivå introducerar Bloom flera nya syntaktiska notationer samtidigt, vilket gör att man kan skriva ner regler för att skriva undantag och förstklassiga grenar. Typsystemet MLPolyR ger full typslutledning , så att författarna övergav utvecklingen av en platt syntax för att explicit skriva typer och stöd för signaturer på modulspråket .
Systemet av Leyen -typ introducerar också en variant av grenpolymorfism: en konstruktion kan representeras som en funktion av högre ordning som tar emot en post från funktioner, som var och en motsvarar en viss beräkningsgren ( Haskells syntax är lämplig för denna ändring och kräver ingen revidering). Till exempel: case
dataList a = noll :: { } | nackdelar :: { hd :: a , tl :: Lista a } snoc xs r = case ( omvänd xs ) r sista xs = snoc xs { noll = \ r -> _ | _ , nackdelar = \ r -> r . hd }Eftersom poster i Layens system är utökningsbara är grenparsning flexibel på nivån av dynamiska beslut (som kedja av kontroller eller användning av en associativ array ), men ger en mycket effektivare implementering (variantetiketten motsvarar en offset i posten). Dock måste standardförgreningsstödet ( default) tas bort i detta fall, eftersom ett enda default-mönster skulle matcha flera fält (och därmed flera förskjutningar). Leyen kallar denna konstruktion " förstklassiga mönster " ( förstklassiga mönster ).
Högre sorts polymorfism betyder abstraktion överordningens typkonstruktorer , det vill säga typoperatorer av formen
* -> * -> ... -> *Stöd för denna funktion tar polymorfism till en högre nivå, vilket ger abstraktion över både typer och typkonstruktorer , precis som funktioner av högre ordning ger abstraktion över både värden och andra funktioner. Polymorfism i högre kön är en naturlig komponent i många funktionella programmeringsspråk , inklusive monader , veck och inbäddningsbara språk . [62] [82]
Till exempel [62] om du definierar följande funktion ( Haskell language ):
när b m = om b så returnerar m annars ()då kommer följande funktionella typ att härledas för det :
när :: forall ( m :: * -> * ) . Monad m => Bool -> m () -> m ()Signaturen m :: * -> *säger att typvariabeln m är en typvariabel som tillhör en högre sort ( engelska högre-kinded typevariabel ). Det betyder att den abstraherar över typkonstruktörer (i detta fall unary , såsom Maybeeller []), som kan appliceras på konkreta typer, såsom Inteller (), för att konstruera nya typer.
På språk som stöder full typabstraktion ( Standard ML , OCaml ) måste alla typvariabler vara av ett släkte * , annars skulle typsystemet vara osäkert . Polymorfism i högre släkten tillhandahålls alltså av själva abstraktionsmekanismen, i kombination med explicit annotering vid instansiering, vilket är något obekvämt. En idiomatisk implementering av polymorfism i högre släkten är dock möjlig, utan att kräva explicit anteckning - för detta används en teknik som liknar defunktionalisering på typnivå . [62]
Snälla system säkerställer själva typsystemens säkerhet genomatttillåta kontroll över betydelsen av typuttryck .
Låt det till exempel krävas att istället för den vanliga typen " vektor " (linjär array) implementera familjen av typer " längdvektorn ", med andra ord för att definiera typen " vektor indexerad efter längd ". Den klassiska Haskell- implementeringen ser ut så här [83] :
data Noll data Succ n data Vec :: * -> * -> * där Noll :: Vec a Zero Cons :: a -> Vec a n -> Vec a ( Succ n )Här definieras fantomtyper [84] först , det vill säga typer som inte har en dynamisk representation. Konstruktörerna Zero och Succfungerar som "typlagervärden", och variabeln nframtvingar olikheten mellan de olika betongtyperna som konstruerats av konstruktören Succ. Typen Vecdefinieras som en generaliserad algebraisk datatyp (GADT).
Lösningen förutsätter villkorligt att fantomtypen nkommer att användas för att modellera vektorns heltalsparameter baserat på Peano-axiomen - det vill säga att endast uttryck som Succ Zero, Succ Succ Zero, Succ Succ Succ Zeroetc. kommer att byggas. Men även om definitionerna är skrivna i typspråk , de själva är formulerade på ett opartat sätt. Detta framgår av signaturen Vec :: * -> * -> *, vilket betyder att de betongtyper som skickas som parametrar tillhör släktet * , vilket betyder att de kan vara vilken betongtyp som helst. Med andra ord, meningslösa typuttryck som Succ Booleller är inte förbjudna här Vec Zero Int. [83]
En mer avancerad kalkyl skulle göra det möjligt att specificera intervallet för typparametern mer exakt:
data Nat = Noll | Succ Nat data Vec :: * -> Nat -> * där VNil :: Vec a Zero VCons :: a -> Vec a n -> Vec a ( Succ n )Men vanligtvis har bara högspecialiserade system med beroende typer [85] implementerade i språk som Agda , Coq och andra en sådan uttrycksförmåga. Till exempel, från Agda- språkets synvinkel, skulle posten Vec :: * -> Nat -> *innebära att könet för en typ Vec beror på typen Nat(det vill säga ett element av ett slag beror på ett element av en annan, lägre sort ).
2012 byggdes en tillägg till Haskell- språket [83] , som implementerar ett mer avancerat system av kön och gör ovanstående kod korrekt Haskell-kod. Lösningen är att alla typer (under vissa restriktioner) automatiskt “ promoteras ” ( eng. promote ) till en högre nivå, och bildar släkten med samma namn som kan användas explicit. Ur denna synvinkel är posten inte beroende - det betyder bara att den andra parametern i vektorn måste tillhöra det namngivna släktet , och i det här fallet är det enda elementet i detta släkte typen med samma namn. Vec :: * -> Nat -> *Nat
Lösningen är ganska enkel, både vad gäller implementering i kompilatorn och vad gäller praktisk tillgänglighet. Och eftersom typpolymorfism i sig är en naturlig del av Haskells semantik, leder typpromotion till typpolymorfism , vilket både ökar kodåteranvändningen och ger en högre nivå av typsäkerhet . Till exempel används följande GADT för att verifiera typlikhet:
data EqRefl a b där Refl :: EqRefl a ahar ett kön i klassisk Haskell , * -> * -> *vilket hindrar den från att användas för att testa för likhet av typkonstruktörer som t.ex. MaybeEtt typfrämjande släktsystem härleder ett polymorft släkte forall X. X -> X -> *, vilket gör typen EqReflmer generisk. Detta kan uttryckligen skrivas:
data EqRefl ( a :: X ) ( b :: X ) där Refl :: forall X . forall ( a :: X ) . EqRefl a aEffektsystem föreslogs av Gifford och Lucassen under andra hälften av 1980 -talet [86] [87] [88] med syftet att isolera bieffekter för bättre kontroll över säkerhet och effektivitet i konkurrenskraftig programmering .
I det här fallet betyder effektpolymorfism kvantifiering över renheten av en specifik funktion, det vill säga införandet av en flagga i den funktionella typen som karakteriserar funktionen som ren eller oren. Denna typtillägg gör det möjligt att abstrahera renheten hos funktioner av högre ordning från renheten hos funktioner som skickas till dem som argument.
Detta är särskilt viktigt när man flyttar till funktioner över moduler ( poster som inkluderar abstrakta typer ) - funktionorer - eftersom de under renhetsförhållanden har rätt att vara applicerande, men i närvaro av biverkningar måste de generera för att säkerställa typsäkerhet (för mer om detta, se motsvarighet av typer i ML-modulspråket ). Sålunda, på språket för högre ordningens förstklassiga moduler, visar sig effektpolymorfism vara en nödvändig bas för att stödja generativitetspolymorfism : att skicka en renhetsflagga till en funktor ger ett val mellan applikativ och generativ semantik i ett enda system. [65]
Typsäker parametrisk polymorfism är tillgänglig på Hindley-Milner -skrivna språk — på ML - dialekter ( Standard ML , OCaml , Alice , F# ) och deras ättlingar ( Haskell , Clean , Idris , Mercury , Agda ) — likaså som i de som ärvts från dem hybridspråk ( Scala , Nemerle ).
Generiska datatyper (generika) skiljer sig från parametriskt polymorfa system genom att de använder en begränsad kvantifiering , och därför inte kan ha en högre rang än 1 . De finns i Ada , Eiffel , Java , C# , D , Rust ; och även i Delphi sedan 2009 version. De dök först upp i CLU .
Intensionell polymorfism är en teknik för att optimera kompileringen av parametrisk polymorfism baserad på komplex typteoretisk analys, som består av beräkningar på typer under körning. Intentionell polymorfism möjliggör taggfri sophämtning , oboxadöverföring av argument till funktioner och inramade (minnesoptimerade) platta datastrukturer. [89] [90] [91]
Monomorfisering är en teknik för att optimera kompileringen av parametrisk polymorfism, som består i att generera en monomorf instans för varje användningsfall av en polymorf funktion eller typ. Med andra ord, parametrisk polymorfism på källkodsnivå översätts till ad hoc- polymorfism på målplattformsnivå. Monomorfisering förbättrar prestandan (mer exakt, gör polymorfism "fri"), men samtidigt kan det öka storleken på utdatamaskinkoden . [92]
I den klassiska versionen är Hindley-Milner-systemet (även helt enkelt "Hindley-Milner" eller "X-M", engelska HM ) [93] [94] som ligger bakom ML -språket en delmängd av System F , begränsat av predikativen prenex polymorfism för att möjliggöra automatisk typinferens , för vilken Hindley-Milner traditionellt också inkluderade den så kallade " Algorithm W " , utvecklad av Robin Milner .
Många implementeringar av X-M är en förbättrad version av systemet, som representerar ett " principiellt typningsschema " [93] [2] som, i ett enda pass med nästan linjär komplexitet, samtidigt härleder de mest allmänna polymorfa typerna för varje uttryck och strikt kontrollerar deras överenskommelse .
Sedan starten har Hindley-Milner-systemet utökats på flera sätt [95] . En av de mest välkända tilläggen är stöd för ad-hoc polymorfism genom typklasser , som ytterligare generaliseras till kvalificerade typer .
Automatisk typinferens ansågs vara en nödvändighet i den ursprungliga utvecklingen av ML som ett interaktivt teorembevisande system " Logic for Computable Functions ", vilket är anledningen till att motsvarande begränsningar infördes. Därefter, på basis av ML , utvecklades ett antal effektivt sammanställda allmänspråk, orienterade mot storskalig programmering , och i detta fall minskar behovet av att stödja typinferens kraftigt, eftersom modulgränssnitt i industriell praxis måste i alla fall uttryckligen annoteras med typer [81] . Därför har många Hindley-Milner-tillägg föreslagits som undviker typinferens till förmån för bemyndigande, upp till och inklusive stöd för ett komplett System F med impredikativ polymorfism, såsom modulspråket av högre ordning , som ursprungligen var baserat på explicit modultypsanteckning och har många tillägg och dialekter, samt Haskell -språktillägg ( , och ). Rank2TypesRankNTypesImpredicativeTypes
Standard ML :s MLton- kompilator monomorfiserar , men på grund av andra optimeringar som är tillämpliga på Standard ML överstiger den resulterande ökningen av utdatakod inte 30 % [92] .
I C är funktioner inte förstklassiga objekt , men det är möjligt att definiera funktionspekare , vilket låter dig bygga funktioner av högre ordning [96] . Osäker parametrisk polymorfism är också tillgänglig genom att uttryckligen skicka de nödvändiga egenskaperna för en typ genom en otypad delmängd av språket som representeras av en otypad pekare ).språkgemenskapeni"pekaregenerisk(kallas en "97][ ad-hoc polymorfism , eftersom den inte ändrar representationen av pekaren, men den är skriven explicit för att kringgå kompilatorns typsystem [96] . void* void*
Till exempel kan standardfunktionen qsorthantera arrayer av element av vilken typ som helst för vilka en jämförelsefunktion [96] är definierad .
struct segment { int start ; int end ; }; int seg_cmpr ( struct segment * a , struct segment * b ) { return abs ( a -> end - a -> start ) - abs ( b -> end - b -> start ); } int str_cmpr ( char ** a , char ** b ) { return strcmp ( * a , * b ); } struct segment segs [] = { { 2 , 5 }, { 4 , 3 }, { 9 , 3 }, { 6 , 8 } }; char * strs [] = { "tre" , "en" , "två" , "fem" , "fyra" }; huvud () { qsort ( strs , sizeof ( strs ) / sizeof ( char * ), sizeof ( char * ), ( int ( * )( void * , void * )) str_cmpr ); qsort ( segs , sizeof ( segs ) / sizeof ( struct segment ), sizeof ( struct segment ), ( int ( * )( void * , void * )) seg_cmpr ); ... }Men i C är det möjligt att idiomatiskt reproducera typad parametrisk polymorfism utan att använda void*[98] .
C++-språket tillhandahåller ett mallundersystem som ser ut som parametrisk polymorfism, men är semantiskt implementerat av en kombination av ad hoc- mekanismer:
mall < typnamn T > T max ( T x , T y ) { om ( x < y ) returnera y ; annan returnera x ; } int main () { int a = max ( 10 , 15 ); dubbel f = max ( 123,11 , 123,12 ); ... }Monomorfisering av vid kompilering av C++-mallar är oundviklig eftersom språkets typsystem saknar stöd för polymorfism - det polymorfa språket här är ett statiskt tillägg till den monomorfa språkkärnan [99] . Detta leder till en multipel ökning av mängden mottagen maskinkod , vilket är känt som " code bloat " [100] .
Notationen av parametrisk polymorfism som en utveckling av lambda-kalkylen (kallad polymorphic lambda-kalkylen eller System F ) beskrevs formellt av logikern Jean-Yves Girard [101] [102] ( 1971 ), oberoende av honom en liknande systemet beskrevs av datavetaren John S. Reynolds [103] ( 1974 ) [104] .
Parametrisk polymorfism introducerades först i ML 1973 [41] [105] . Oberoende av honom implementerades parametriska typer under ledning av Barbara Liskov vid CLU ( 1974 ) [41] .
Datatyper | |
---|---|
Otolkbart | |
Numerisk | |
Text | |
Referens | |
Sammansatt | |
abstrakt | |
Övrig | |
Relaterade ämnen |