To understand rest parameters in JavaScript, let’s start with creating a function to add two numbers.
function addNumbers(num1, num2) {
let res = num1 + num2;
return res;
}
let result = addNumbers(7, 2);
console.log(result); // 9
The above function takes two input arguments and returns the addition of passed arguments. So, if you pass two numbers, the addNumbers function would return the sum of those two numbers.
However, by design all JavaScript functions are overloaded, which means you can pass:
In the definition of the overloaded function, you can ignore types, as JavaScript is a type-less language. Let’s pass fewer arguments to the addNumbers function:
function addNumbers(num1, num2) {
let res = num1 + num2; // 7 + undefined
return res;
}
let result = addNumbers(7);
console.log(result); // NaN
As you see, JavaScript does not complain about fewer arguments and returns NaN as a result. Next, pass more arguments to the addNumbers function:
function addNumbers(num1, num2) {
let res = num1 + num2;
return res;
}
let result = addNumbers(7, 2, 4, 9, 1, 6);
console.log(result); // 9
Once again, JavaScript does not complain about extra arguments, and it assigns the first two arguments to num1 and num2 and returns the addition of them.
Now, you may have a couple of questions regarding extra arguments to the function:
The answer to the above questions is rest parameters. JavaScript does not ignore the extra arguments and represents them inside the function using the rest parameters.
Rest parameters represent an unknown number of arguments as an array inside the function.
Some characteristics of the rest parameters are:
• It represents the extra arguments passed in the function.
• It is always defined as the last formal parameter of the function.
• It is always prefixed with three dots. A rest parameter with name “foo” is represented by …foo.
In the addNumbers function, a rest parameter can be added as shown next in the next code listing:
function addNumbers(num1, num2, ...ep) {
console.log(ep.length); // 4
let res = num1 + num2;
return res;
}
let result = addNumbers(7, 2, 4, 9, 1, 6);
console.log(result); // 9
Now, the addNumbers function has a rest parameter with name ep. You can give any name to the rest parameter. As the rest parameters represent extra arguments passed to the function, you get 4 as the output on printing the length of it.
The rest parameter is an ** array object**, which means you can perform operations such as push, pop, length, sort, map, etc. You can perform various array operations on the rest parameter as shown in the next code listing:
function addNumbers(num1, num2, ...ep) {
console.log(ep.length); // 4
ep.push(9);
ep[0] = 10;
console.log(ep.length); // 5
console.log(ep.sort()); // [1,10,6,9,9]
let res = num1 + num2;
return res;
}
let result = addNumbers(7, 2, 4, 9, 1, 6);
console.log(result); // 9
One important thing you should keep in mind is that the rest parameter must always be the last formal parameter of the function. The code listed below throws an error:
function addNumbers(num1,...ep,num2){
let res = num1+ num2;
return res;
}
For the above function, JavaScript should throw an error that the Rest parameter must be the last formal parameter.
You can use the normal parameter and rest parameter together to achieve any particular requirement. In the next example, the function multiplies all the extra parameters with the formal parameter and returns the array.
function multiplyNumbers(num, ...ep) {
let res = ep.map(e => {
return num * e;
});
return res;
}
let result = multiplyNumbers(7, 2, 4, 9, 1, 6);
console.log(result); // [14,18,63,7,42]
Besides rest parameters, you can also use the arguments object to work with extra arguments passed to a function. Each time you call a function, JavaScript passes the arguments object in it.
function addNumbers(num1,num2){
console.log(arguments.length); // 5
let res = num1+ num2;
return res;
}
let res = addNumbers(7,5,8,9,2);
Unlike rest parameters, the arguments object contains all arguments passed to the function—that is why you are getting 5 on printing the length of the arguments object. Some of the significant differences between rest parameters and the arguments object are as follows:
So, for the following code, JavaScript will throw an error:
function addNumbers(num1,num2){
console.log(arguments.length); // 5
arguments.sort();
arguments.pop();
let res = num1+ num2;
return res;
}
let res = addNumbers(7,5,8,9,2);
Each function has its own arguments object, so if you want to access the arguments object of the parent function inside a child function, it is not directly possible. As you see in the code listed below, the display function and addNumbers function both have an arguments object.
function addNumbers(num1,num2){
console.log(arguments.length); // 5
console.log(arguments[0]); // 7
function display(){
// display function has its own arguments object
console.log(arguments.length); //2
console.log(arguments[0]); //1
}
display(1,9);
}
let res = addNumbers(7,5,8,9,2);
However, a child function can directly access the rest parameter of the parent function, as shown in the code listed below:
function addNumbers(num1,num2,...ep){
console.log(ep.length); // 3
function display(){
console.log(ep.length); //3
}
display(1,9);
}
let res = addNumbers(7,5,8,9,2);
So, when it comes to working with nested functions, rest parameters are a better option than an arguments object.
In summary, you should know the following points about rest parameters:
I hope you find this article useful. Thanks for reading.
Dhananjay Kumar is an independent trainer and consultant from India. He is a published author, a well-known speaker, a Google Developer Expert, and a 10-time winner of the Microsoft MVP Award. He is the founder of geek97, which trains developers on various technologies so that they can be job-ready, and organizes India's largest Angular Conference, ng-India. He is the author of the best-selling book on Angular, Angular Essential. Find him on Twitter or GitHub.