Another way to invoke functions in JavaScript is using the .apply()
method. The syntax for .apply()
is as follows: functionName.apply(thisArg, [array])
. "functionName
" refers to the function being called. However, with .apply()
, an array containing all values can be sent instead of individual arguments. Additionally, "thisArg
" specifies the object that "this" references inside the invoked function.
Examples
Using apply() to append an array to another
You can use Array.prototype.push()
to append an element to an array. Because push()
accepts a variable number of arguments, you can also push multiple elements at once. But if you pass an array to push()
,
it will actually add that array as a single element, instead of adding
the elements individually, ending up with an array inside an array. On
the other hand, Array.prototype.concat()
does have the desired behavior in this case, but it does not append to the existing array - it creates and returns a new array.
In this case, you can use apply
to implicitly "spread" an array as a series of arguments.
const array = ["a", "b"]; const elements = [0, 1, 2]; array.push.apply(array, elements); console.info(array); // ["a", "b", 0, 1, 2]
The same effect can be achieved with the spread syntax.
const array = ["a", "b"]; const elements = [0, 1, 2]; array.push(...elements); console.info(array); // ["a", "b", 0, 1, 2]
Using apply() and built-in functions
Clever usage of apply()
allows you to use built-in functions for some tasks that would probably
otherwise require manually looping over a collection (or using the
spread syntax).
For example, we can use Math.max()
and Math.min()
to find out the maximum and minimum value in an array.
// min/max number in an array const numbers = [5, 6, 2, 3, 7]; // using Math.min/Math.max apply let max = Math.max.apply(null, numbers); // This about equal to Math.max(numbers[0], …) // or Math.max(5, 6, …) let min = Math.min.apply(null, numbers); // vs. simple loop based algorithm max = -Infinity; min = +Infinity; for (let i = 0; i < numbers.length; i++) { if (numbers[i] > max) { max = numbers[i]; } if (numbers[i] < min) { min = numbers[i]; } }
But beware: by using apply()
(or the spread syntax) with
an arbitrarily long arguments list, you run the risk of exceeding the
JavaScript engine's argument length limit.
The consequences of calling a function with too many arguments (that
is, more than tens of thousands of arguments) is unspecified and varies
across engines. (The JavaScriptCore engine has a hard-coded argument limit of 65536.)
Most engines throw an exception; but there's no normative specification
preventing other behaviors, such as arbitrarily limiting the number of
arguments actually passed to the applied function. To illustrate this
latter case: if such an engine had a limit of four arguments (actual
limits are of course significantly higher), it would be as if the
arguments 5, 6, 2, 3
had been passed to apply
in the examples above, rather than the full array.
If your value array might grow into the tens of thousands, use a hybrid strategy: apply your function to chunks of the array at a time:
function minOfArray(arr) { let min = Infinity; const QUANTUM = 32768; for (let i = 0; i < arr.length; i += QUANTUM) { const submin = Math.min.apply( null, arr.slice(i, Math.min(i + QUANTUM, arr.length)), ); min = Math.min(submin, min); } return min; } const min = minOfArray([5, 6, 2, 3, 7]);