Anonimna funkcija
U programiranju, anonimna funkcija je funkcija definisana tako da nije vezana za identifikator. Anonimne funkcije su često:[1]
- argumenti prosleđeni funkcijama višeg reda, ili
- se koriste za konstruisanje rezultata funkcije višeg reda koja treba da vrati funkciju.
Ako se funkcija koristi jednom, ili ograničen broj puta, anonimna funkcija može biti sintaksički "lakša" od korišćenja imenovane funkcije. Anonimne funkcije su česte u jezicima funkcionalnog programiranja i jezicima koji podržavaju funkcije prve klase (first-class funkcije), gde ispunjavaju ulogu tipa funkcije kao što literali rade za tipove podataka.
Anonimne funkcije je uveo Alonzo Čerč kada je otkrio lambda račun 1936. godine, pre nastanka elektronskih računara, kada su sve funkcije bile anonimne.[2] U nekoliko programskih jezika, anonimne funkcije su predstavljene ključnom rečju lambda, i anonimne funkcije se često nazivaju lambdama ili lambda apstrakcijama. Anonimne funkcije su deo programskih jezika još od nastanka programskog jezika Lisp 1958. i sve veći broj modernih programskih jezika podržava anonimne funkcije. Takođe anonimne funkcije se mogu posmatrati kao forma ugnježdenih funkcija.
Upotreba
[уреди | уреди извор]Anonimne funkcije mogu biti korišćene za funkcionalnost koja ne treba biti imenovana i za kratkotrajnu upotrebu. Neki bitniji primeri su u korišćenju closure i currying funkcija.
Korišćenje anonimnih funkcija je pitanje stila. Njihovo korišćenje nikada nije jedini način za rešavanje problema; Svaka anonimna funkcija može biti definisana imenovanom funkcijom i biti pozivana imenom. Neki programeri koriste anonimne funkcije za pisanje kodova koji se neće izvršavati više puta, a u sebi sadrži veliki broj jednolinijskih funkcija.
U nekim programskim jezicima, anonimne funkcije se često implementiraju za veoma specifične potrebe kao što je korišćenje u obliku callback funkcija, ili instanciranje funkcije za konkretne vrednosti, koje mogu biti efikasnije, čitljivije i manje sklone greškama nego imenovane funkcije.
Kod u sledećim primerima je napisan u jeziku Python 2.x .
Sortiranje
[уреди | уреди извор]Kad pokušavamo da sortiramo nestandardnim načinom, korišćenje anonimne funkcije u komparacione svrhe može biti jednostavnije od korišćenja imenovanih funkcija. Većina jezika nudi generičku funkciju koja implementira algoritam za sortiranje koji će sortirati proizvoljne objekte. Ove funkcije obično primaju proizvolju funkciju poređenja koja prima dva proizvoljna objekta i vraća vrednosti manje od 0, 0 ili veće od 0 u odnosu na to da li je drugi prosleđen objekat veći, jednak ili manji od prvog prosleđenog.
Sortiranje liste stringova po njihovoj dužini:
>>> a = ['house', 'car', 'bike']
>>> a.sort(lambda x,y: cmp(len(x), len(y)))
>>> print(a)
['car', 'bike', 'house']
Anonimna funkcija u ovom primeru je lambda izraz:
lambda x,y: cmp(...)
Anonimna funkcija prima dva argumenta, x i y, i vraća poređenje između njih koristeći ugrađenu funkciju cmp().
>>> a = [10, 'number', 11.2]
>>> a.sort(lambda x,y: cmp(x.__class__.__name__, y.__class__.__name__))
>>> print(a)
[11.2, 10, 'number']
Closure
[уреди | уреди извор]Closure funkcije su funkcije koje imaju vrednost u okruženju koje sadrži 'bound' promenljive.
Sledeći primer povezuje promenljivu "threshold" sa anonimnom funkcijom koja poredi ulaz sa njom.
def comp(threshold):
return lambda x: x < threshold
Ovo može služiti kao generator funkcija poređenja.
>>> func_a = comp(10)
>>> func_b = comp(20)
>>> print func_a(5), func_a(8), func_a(13), func_a(21)
True True False False
>>> print func_b(5), func_b(8), func_b(13), func_b(21)
True True True False
Bilo bi nepraktično kreirati funkciju za svaku funkciju poređenja i može biti nepraktično čuvati vrednost po kojoj se poredi za dalju upotrebu. I pored razloga zašto se closure funkcije koriste, anonimne funkcije su entitet koji sadrži funkcionalnost koja vrši poređenje.
Currying
[уреди | уреди извор]Currying je proces promene funkcije da bi bilo potrebno manje ulaznih vrednosti (u ovom slučaju transformacija funkcije koja izvodi deljenje bilo kog celog broja u onu funkciju koja vrši deljenje sa skupom celih brojeva).
>>> def divide(x, y):
... return x / y
>>> def divisor(d):
... return lambda x: divide(x, d)
>>> half = divisor(2)
>>> third = divisor(3)
>>> print half(32), third(32)
16 10
>>> print half(40), third(40)
20 13
Korišćenje anonimnih funkcija nije česta kod currying tehnike, ali se može koristiti u te svrhe.U primeru iznad, funkcija divisor generiše funkciju sa konkretnim deliteljem.Funkcija half i third izvodi funkciju divide sa fiksiranim deliteljem.
Funkcija divisor takođe primenjuje clousure tehniku tako što vezuje promenljivu "d".
Funkcije višeg reda
[уреди | уреди извор]Python 2.x sadrži nekoliko funkcija koje primaju anonimne funkcije kao argument.Ovaj odeljak opisuje njih.
Map
[уреди | уреди извор]Map funkcija vrši poziv funkcije nad svakim elementom liste. Sledeći primer kvadrira svaki element u nizu anonimnom funkcijom.
>>> a = [1, 2, 3, 4, 5, 6]
>>> print map(lambda x: x*x, a)
[1, 4, 9, 16, 25, 36]
Anonimna funkcija prima argument i množi ga sobom (kvadrira ga). Po mišljenju kreatora jezika, forma iznad nije preporučljiva, vec predlažu sledeću formu sa istim značenjem a bolje se uklapa sa namenom jezika:
>>> a = [1, 2, 3, 4, 5, 6]
>>> print [x*x for x in a]
[1, 4, 9, 16, 25, 36]
Filter
[уреди | уреди извор]Filter funkcija vraća sve elemente liste koje imaju vrednost True kad ih prosledimo određenoj funkciji.
>>> a = [1, 2, 3, 4, 5, 6]
>>> print filter(lambda x: x % 2 == 0, a)
[2, 4, 6]
Anonimna funkcija proverava da li je prosleđeni argument paran. Isto kao u slučaju sa mapom, kod ispod je preporučljiviji.
>>> a = [1, 2, 3, 4, 5, 6]
>>> print [x for x in a if x % 2 == 0]
[2, 4, 6]
Fold
[уреди | уреди извор]Fold/reduce funkcija prolazi kroz sve elemente liste (obično s leva na desno), i nagomilava vrednost pri tom prolasku. Česta upotreba ovoga je kombinovanje elemenata liste u jednu vrednost. Na primer:
>>> a = [1, 2, 3, 4, 5]
>>> print reduce(lambda x,y: x*y, a)
120
Ovo izvršava sledeće:
Anonimna funkcija ovde je množenje dva argumenta.
Rezultat fold funkcije mora biti jedna vrednost. Umesto toga, i map i filter mogu biti kreirani korišćenjem fold funkcije.U mapi, vrednost koja je nagomilana je zapravo nova lista koja sadrži rezultate primene funkcije na svaki element originalne liste. U filter funkciji, vrednost koja je nagomilana je nova lista koja sadrži samo one elemente koji ispunjavaju određen uslov.
Spisak jezika
[уреди | уреди извор]Sledeća lista programskih jezika neimenovane anonimne funkcije podrzava u potpunosti, delom, ili ne podržava.
Ova tabela ukazuje ne neke opšte trendove. Jezici koji ne podržavaju anonimne funkcije(C, Pascal, Object Pascal) su konvencijalni strogo tipizirani jezici. Medjutim, strogo tipizirani jezici mogu podržavati anonimne funkcije.Na primer, ML jezici su strogo tipizirani i sadrže anonimne funkcije, i Delphi, kao dijalekt objektnog Pascal-a je nadograđen tako da podržava anonimne funkcije.Takođe, jezici koji tretiraju funkcije kao 'funkcije prvog reda'(Dylan, Haskell, JavaScript,Lisp,ML, Perl,Python, Ruby,Scheme) sadrže anonimne funkcije tako da mogu biti definisane i prosleđene jednostavno kao i tipovi podataka. Međutim, novi C++11 standard ih dodaje C++-u iako je konvencijalan, strogo tipiziran jezik.
Language | Support | Notes |
---|---|---|
ActionScript | ||
Ada | 'Expression' funkcije su deo jezika Ada2012 | |
ALGOL 68 | ||
Brainfuck | ||
Bash | Biblioteka je napravljena tako da sadrži anonimne funkcije u jeziku Bash. | |
C | Podrška je omogućena u Clang-u zajedno sa LLVM comiler-rt bibliotekom. GCC podrška je data za makro implementaciju koja omogućava upotrebu. Detalje videti ispod. | |
C#] | ||
C++ | Podrška postoji od C++11 standarda. | |
CFML | Podrška postoji od Railo 4 ColdFusion 10. | |
Clojure | ||
COBOL | Micro Focus-ov nestandardni Managed COBOL dijalekt podržava lambde, koje se nazivaju anonimnim metodama. | |
Curl | ||
D | ||
Dart | ||
Delphi | ||
Dylan | ||
Eiffel | ||
Elixir | ||
Erlang | ||
F# | ||
Factor | "Quotations" podržava ovo. | |
Fortran | ||
Frink | ||
Go | ||
Gosu | [3] | |
Groovy | [4] | |
Haskell | ||
Haxe | ||
Java | Podržano u Java 8. videti Java ograničenja ispod za detalje. | |
JavaScript | ||
Julia | ||
Lisp | ||
Logtalk | ||
Lua | ||
MUMPS | ||
Mathematica | ||
Maple | ||
MATLAB | ||
Maxima | ||
OCaml | ||
Octave | ||
Object Pascal | Delfi, dijalekt Objektnog paskala podržava anonimne funkcije još od Delphi-ja 2009. Oxygene Object Pascal dijalekt ih takođe podržava. | |
Objective-C (Mac OS X 10.6+) | ||
Pascal | ||
Perl | ||
PHP | Od verzije PHP 5.3.0 prave anonimne funkcije su podržane. Pre toga, bile su podržane samo delom, koji je radio slično kao C#-ova imprementacija. | |
PL/I | ||
Python | Python podržava anonimne funkcije kroz lambda sintaksu, koja podržava samo izraze, ne naredbe. | |
R | ||
Racket | ||
Rexx | ||
RPG | ||
Ruby | Ruby-jeve anonimne funkcije su nasleđene od Smalltalk-a, nazivaju se 'blokovi'. | |
Rust | ||
Scala | ||
Scheme | ||
Smalltalk | Smalltalk-ove anonimne funkcije nazivaju se 'blokovi'. | |
Standard ML | ||
Swift | Swift-ove anonimne funkcije se nazivaju 'Closures'. | |
TypeScript | ||
Tcl | ||
Vala | ||
Visual Basic .NET v9 | ||
Visual Prolog v 7.2 | ||
Wolfram Language |
Референце
[уреди | уреди извор]- ^ "Higher order functions". learnyouahaskell.com. Retrieved 3 December 2014.
- ^ Fernandez 2009, стр. 33.
- ^ „Gosu Documentation” (PDF). Архивирано из оригинала (PDF) 01. 05. 2013. г. Приступљено 4. 3. 2013.
- ^ „Groovy Documentation”. Архивирано из оригинала 22. 05. 2012. г. Приступљено 29. 5. 2012.
Литература
[уреди | уреди извор]- Fernandez, Maribel (2009). Models of Computation: An Introduction to Computability Theory. Springer Science & Business Media. стр. 33. ISBN 978-1-84882-434-8.