最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - Finding maximum value in array recursively - Stack Overflow

programmeradmin2浏览0评论

I am trying to use recursion to find the maximum value in an array in JavaScript. I created the function, but cannot figure out how to do it recursively.

function Max(a) {
  var a = [2,3,5];
  return Math.max.apply(Math, a);
}

I am trying to use recursion to find the maximum value in an array in JavaScript. I created the function, but cannot figure out how to do it recursively.

function Max(a) {
  var a = [2,3,5];
  return Math.max.apply(Math, a);
}
Share Improve this question edited Nov 5, 2015 at 6:20 user3717023 asked Nov 5, 2015 at 0:44 littlemisschiklittlemisschik 1391 gold badge3 silver badges9 bronze badges 6
  • 4 Recursion is the function calling itself. If the function gets passed the array of values, what is it that you want to pass to the inner function call? – Felix Kling Commented Nov 5, 2015 at 0:46
  • 1 Hint: max(a,b,c) = max(max(a,b),c). – mpen Commented Nov 5, 2015 at 0:58
  • stackoverflow./questions/1379553/… – Tom Netzband Commented Nov 5, 2015 at 1:00
  • 1 Why are the array values set inside your function? – nnnnnn Commented Nov 5, 2015 at 1:01
  • 1 The Math.max function does the recursive (or iterative?) part "under the hood" in the browser's core code. So if your goal/assignment is to write an "iterative function" you should maybe look here (wrong language, right idea). – James Commented Nov 5, 2015 at 1:06
 |  Show 1 more ment

11 Answers 11

Reset to default 2
let max = (list) => {

  // Returns first number if list length is 1
  if (list.length === 1) return list[0]

  // Returns the greater number between the first 2 numbers in an array
  if (list.length === 2) return (list[0] > list[1]) ? list[0] : list[1]

  // If array length is 3+ (Read Below)
  const subMax = max(list.slice(1));
  return (list[0] > subMax) ? list[0] : subMax;
}

// Example
max([5,5,5,8,6,7,4,7])

It is important to understand how the 'Call Stack' works for recursions.

In the example above, since the array length is over 3 the first line to be called is the

const subMax = max(list.slice(1));

All this line does is take out the first item in the array and recalls the function with the shorter array as its argument. This continues until the arrays length is 2 then returns the greater of the 2. Then the function can plete all its previous partially pleted states.

ES6 syntax makes this really concise.

const findMax = arr => {
  if (!Array.isArray(arr)) throw 'Not an array'
  if (arr.length === 0) return undefined
  const [head, ...tail] = arr
  if (arr.length === 1) return head
  return head > findMax(tail) 
    ? head
    : findMax(tail)
}

Here is an example of a recursive function that takes an array of numbers and pares the first two, removes the lesser, and then calls itself again on the given array minus the removed number.

function max(numArray) 
{
    // copy the given array 
    nums = numArray.slice();

    // base case: if we're at the last number, return it
    if (nums.length == 1) { return nums[0]; }

    // check the first two numbers in the array and remove the lesser
    if (nums[0] < nums[1]) { nums.splice(0,1); }
    else { nums.splice(1,1); }

    // with one less number in the array, call the same function
    return max(nums);
}

Here's a jsfiddle: https://jsfiddle/t3q5sm1g/1/

This is actually one of the questions I’ve asked candidates to a software position: Find the maximum number in a jagged array. Each element of the array can be a number or an array of other numbers on indefinite number of levels, like this:

var ar = [2, 4, 10, [12, 4, [100, 99], 4], [3, 2, 99], 0];

And now, just for fun and to keep this post short, I will include two solutions to the question. The first solution uses recursion while the second solution uses a stack. As a matter of fact all these problems with recursion can be also solved by using the stack approach.

Solution I: Find maximum using recursion

// Use recursion to find the maximum numeric value in an array of arrays
function findMax1(ar)
{
    var max = -Infinity;

    // Cycle through all the elements of the array
    for(var i = 0; i < ar.length; i++)
    {
        var el = ar[i];

        // If an element is of type array then invoke the same function
        // to find out the maximum element of that subarray
        if ( Array.isArray(el) )
        {
            el = findMax1( el );
        }

        if ( el > max )
        {
            max = el;
        }
    }

    return max;
}

Solution II: Find maximum using a stack

// Use a stack to find the maximum numeric value in an array of arrays
function findMax2(arElements)
{
    var max = -Infinity;

    // This is the stack on which will put the first array and then 
    // all the other sub-arrays that we find as we traverse an array     
    var arrays = [];

    arrays.push(arElements);

    // Loop as long as are arrays added to the stack for processing
    while(arrays.length > 0)
    {
        // Extract an array from the stack
        ar = arrays.pop();

        // ... and loop through its elements
        for(var i = 0; i < ar.length; i++)
        {
            var el = ar[i];

            // If an element is of type array, we'll add it to stack
            // to be processed later
            if ( Array.isArray(el) )
            {
                arrays.push(el);
                continue;
            }

            if ( el > max )
            {
                max = el;
            }
        }
    }

    return max;
}

Feel free to optimize the above code. You can also find this code as a gist on github.

function max(array) {
  if (array.length === 1) {  // Step1: set up your base case
      return array[0]
 } else {  
     return Math.max(array.shift(), max(array); // Step2: rec case
 }
}

Each time the recursive case is called it gets it closer to the base case.

Math.max accepts two numbers, then pares them, then returns the higher out of the two.

Each time you call array.shift() you are popping off the first element in the array from the array, so the second argument in the recursive call is the array shortened by one.

When array.length is down to just one element, return that element and watch the stack unfold.

Try this:

const biggestElement = list => {
    if (list.length == 1)
        return list[0];
    if (list[0] > list[1])
        list[1] = list[0];
    list = list.splice(1);
    return biggestElement(list);
}

    function recursiveMaxNumber(items){ 

       if(items.length === 1){
         return items[0] //Base-case reached
          }
      else{
            if(items[0]>=items[items.length-1]){
                items.pop() //remove the last item in the array
                return recursiveMaxNumber(items) 
            }
            else{
                return recursiveMaxNumber(items.slice(1)) //remove the first item in the array
            }
         }
    }
    
    console.log(recursiveMaxNumber([2,22,3,3,4,44,33]))

Though its not an efficient way below is my try,

function max(a, b) {
  return a > b ? a : b
}

function findMaxOfArrayRecursion(arr) {
  if (arr.length == 1) {
    return arr[0]
  }
  return max(arr[0], findMaxOfArrayRecursion(arr.slice(1)))
}

Basically we are assuming the first array element to be the maximum & paring it with other elements in a recursive way.

Adding an efficient non-recursive approach as well just for reference,

Math.max(...arr)

const getMax = (arr) => {
    // base case
    if(arr.length === 0) return -Infinity;
    if(arr.length === 1) return arr[0];

    // recursive case
    return Math.max(arr[0], getMax(arr.slice(1)));
}

getMax([5, 2, 11, 9])   //11
function max(arr) {
  if (arr.length > 0) {
    return arr[arr.length - 1] > max(arr.slice(0, -1))
      ? arr[arr.length - 1]
      : max(arr.slice(0, -1));
  }
  return 0;
}

max([3, 2, 1, 7, 5]) //7

or with a bit reacher naming:

function max(arr) {
  if (arr.length > 0) {
    const currentValue = arr[arr.length - 1];
    const nextValue = max(arr.slice(0, -1));

    return currentValue > nextValue ? currentValue : nextValue;
  }
  return 0;
}

I just want to add another possibility without using slice/splice method, but still simple:

function searcMaxValue(numbers, max = 0, index = 0) {
    let actual = numbers[index] | 0;
    if (actual > max) max = actual;

    if (index >= numbers.length - 1) return max;
    return searcMaxValue(numbers, max, ++index);
}
发布评论

评论列表(0)

  1. 暂无评论