|
| 1 | +## `arguments`-olio |
| 2 | + |
| 3 | +Jokainen JavaScriptin näkyvyysalue pääsee käsiksi erikoismuuttujaan nimeltään `arguments`. Tämä muuttuja sisältää listan kaikista funktiolle annetuista argumenteista. |
| 4 | + |
| 5 | +> **Huomio:** Mikäli `arguments` on jo määritelty funktion sisällä joko näkyvyysalueen, `var`-lauseen tai parametrin kautta, `arguments`-oliota ei luoda. |
| 6 | +
|
| 7 | +`arguments`-olio **ei** ole `Array`. Sen semantiikka, erityisesti `length`-ominaisuus, muistuttaa taulukkoa. Tästä huolimatta se ei peri `Array.prototype`:stä ja on itse asiassa `Object`. |
| 8 | + |
| 9 | +Tästä johtuen `arguments`-olioon **ei** voida soveltaa normaaleja taulukkometodeja, kuten `push`, `pop` tai `slice`. Vaikka iterointi onnistuukin `for`-luuppeja käyttäen, tulee se muuttaa aidoksi `Array`-olioksi ennen kuin siihen voidaan soveltaa näitä metodeja. |
| 10 | + |
| 11 | +### Array-olioksi muuttaminen |
| 12 | + |
| 13 | +Alla oleva koodi palauttaa uuden `Array`-olion, joka sisältää `arguments`-olion kaikki jäsenet. |
| 14 | + |
| 15 | + Array.prototype.slice.call(arguments); |
| 16 | + |
| 17 | +Tämä muutos on luonteeltaan **hidas** eikä sitä suositella käytettävän suorituskykyä vaativissa osissa koodia. |
| 18 | + |
| 19 | +### Argumenttien antaminen |
| 20 | + |
| 21 | +Funktiosta toiselle voidaan antaa argumentteja seuraavasti. |
| 22 | + |
| 23 | + function foo() { |
| 24 | + bar.apply(null, arguments); |
| 25 | + } |
| 26 | + function bar(a, b, c) { |
| 27 | + // tee jotain |
| 28 | + } |
| 29 | + |
| 30 | +Toinen keino on käyttää sekä `call`- että `apply`-funktioita yhdessä ja luoda nopeita, sitomattomia kääreitä. |
| 31 | + |
| 32 | + function Foo() {} |
| 33 | + |
| 34 | + Foo.prototype.method = function(a, b, c) { |
| 35 | + console.log(this, a, b, c); |
| 36 | + }; |
| 37 | + |
| 38 | + // Luo "metodin" sitomaton versio |
| 39 | + // Se ottaa seuraavat parametrit: this, arg1, arg2...argN |
| 40 | + Foo.method = function() { |
| 41 | + |
| 42 | + // Tulos: Foo.prototype.method.call(this, arg1, arg2... argN) |
| 43 | + Function.call.apply(Foo.prototype.method, arguments); |
| 44 | + }; |
| 45 | + |
| 46 | + |
| 47 | +### Muodolliset parametrit ja argumenttien indeksit |
| 48 | + |
| 49 | +`arguments`-olio luo sekä *getter*- että *setter*-funktiot sekä sen ominaisuuksille että myös funktion muodollisille parametreille. |
| 50 | + |
| 51 | +Tästä seuraa, että muodollisen parametrin arvon muuttaminen muuttaa myös `arguments`-olion vastaavan ominaisuuden arvoa ja toisin päin. |
| 52 | + |
| 53 | + function foo(a, b, c) { |
| 54 | + arguments[0] = 2; |
| 55 | + a; // 2 |
| 56 | + |
| 57 | + b = 4; |
| 58 | + arguments[1]; // 4 |
| 59 | + |
| 60 | + var d = c; |
| 61 | + d = 9; |
| 62 | + c; // 3 |
| 63 | + } |
| 64 | + foo(1, 2, 3); |
| 65 | + |
| 66 | +### Suorituskykyyn liittyviä myyttejä ja totuuksia |
| 67 | + |
| 68 | +`arguments`-olio luodaan aina paitsi jos se on jo julistettu nimenä funktiossa tai sen muodollisena parametrina. Tämä siitä huolimatta käytetäänkö sitä vai ei. |
| 69 | + |
| 70 | +Sekä *getter*- ja *setter*-funktiot luodaan **aina**. Tästä seuraa, että niiden käytöllä ei ole juurikaan merkitystä suorituskyvyn kannalta. |
| 71 | + |
| 72 | +> **ES5 Huomio:** Näitä *getter*- ja *setter*-funktioita ei luoda tiukassa moodissa. |
| 73 | +
|
| 74 | +On kuitenkin eräs tapaus, jossa suorituskyky kärsii. Tämä liittyy `arguments.callee`-ominaisuuden käyttöön. |
| 75 | + |
| 76 | + function foo() { |
| 77 | + arguments.callee; // tee jotain tällä funktio-oliolla |
| 78 | + arguments.callee.caller; // ja kutsuvalla funktio-oliolla |
| 79 | + } |
| 80 | + |
| 81 | + function bigLoop() { |
| 82 | + for(var i = 0; i < 100000; i++) { |
| 83 | + foo(); // normaalisti tämä olisi inline-optimoitu |
| 84 | + } |
| 85 | + } |
| 86 | + |
| 87 | +Yllä olevassa koodissa `foo`-kutsua ei voida [käsitellä avoimesti][1], koska sen tulee tietää sekä itsestään että kutsujasta. Sen lisäksi, että se haittaa suorituskykyä, rikkoo se myös kapseloinnin. Tässä tapauksessa funktio voi olla riippuvainen tietystä kutsuympäristöstä. |
| 88 | + |
| 89 | +On **erittäin suositeltavaa** ettei `arguments.callee`-ominaisuutta tai sen ominaisuuksia käytetä **ikinä**. |
| 90 | + |
| 91 | +> **ES5 Huomio:** Tiukassa moodissa `arguments.callee` palauttaa `TypeError`-virheen, koska se käyttö on vanhennettu. |
| 92 | +
|
| 93 | +[1]: http://en.wikipedia.org/wiki/Inlining |
| 94 | + |
0 commit comments