Search code examples
javascriptcountmergesortinversion

Javascript implementation of the inversion-counting with merge-sort algorithm


i am trying to implement the inversion-counting using merge sort algorithm in javascript. I found description and pseudo-code on this site. My implementation looks like this:

var mergeAndCount, sortAndCount;

/*
the merging routine
@param List1 the first list to be merged
@param List2 the second list to be merged
*/

mergeAndCount = function(List1, List2) {
  var count, outputList;
  outputList = [];
  count = 0;
  while (List1.length > 0 || List2.length > 0) {
    outputList.push(Math.min(List1[0], List2[0]));
    if (List2[0] < List1[0]) {
      count += List1.length;
      List2.shift();
    } else {
      List1.shift();
    }
  }

  outputList = outputList.concat(List1.concat(List2));
  return {
    'count': count,
    'list': outputList
  };
};

/*
count inversion algorithm
@param List the sequence to be sorted
*/
sortAndCount = function(List) {
  var List1, List2, mergeOut, output1, output2;
  if (List.length < 2) {
    return {
      'count': 0,
      'list': List
    };
  } else {
    List1 = List.splice(0, List.length / 2);
    List2 = List;
    output1 = sortAndCount(List1);
    output2 = sortAndCount(List2);
    mergeOut = mergeAndCount(List1, List2);
    return {
      'count': output1.count + output2.count + mergeOut.count,
      'list': mergeOut.list
    };
  }
};

I wanted to test it on Jsfiddle here, but it crashes (too much memory used). Somehow it works for the inupt [1, 3, 2] but not for other. I am not sure what is going wrong, if my implementation or the original pseudocode is false.


Solution

  • Error 1 : infinite loop

    The while goes on for a very long time when it starts to compare numbers with undefined. If List1.length is 0, the comparison List2[0] < List1[0] will always be false, resulting in List1.shift() which changes nothing.

    Replace:

    while (List1.length > 0 || List2.length > 0) {
    

    With:

    while (List1.length > 0 && List2.length > 0) {
    

    Error 2 : manipulating arrays

    You alter the arrays and then use what you expect to be their initial values. At the begining of each function you should copy the arrays (using slice is the fastest way).

    Error 3 : ignoring output of sortAndCount

    Replace:

    mergeOut = mergeAndCount(List1, List2);
    

    With:

    mergeOut = mergeAndCount(output1.list, output2.list);  
    

    Correct solution:

    var mergeAndCount, sortAndCount;
    
    /*
    the merging routine
    @param List1 the first list to be merged
    @param List2 the second list to be merged
    */
    
    mergeAndCount = function(List1, List2) {
      List1 = List1.slice();
      List2 = List2.slice();
      var count = 0;
      var outputList = [];
    
      while (List1.length > 0 && List2.length > 0) {
        outputList.push(Math.min(List1[0], List2[0]));
        if (List2[0] < List1[0]) {
          count += List1.length;
          List2.shift();
        } else {
          List1.shift();
        }
      }
      outputList = outputList.concat(List1.concat(List2));
      return {
        'count': count,
        'list': outputList
      };
    };
    
    /*
    count inversion algorithm
    @param List the sequence to be sorted
    */
    sortAndCount = function(List) {
      List = List.slice();
      var List1, List2, mergeOut, output1, output2;
      if (List.length < 2) {
        return {
          'count': 0,
          'list': List
        };
      } else {
        List1 = List.splice(0, Math.floor(List.length / 2));
        List2 = List;
        output1 = sortAndCount(List1);
        output2 = sortAndCount(List2);
        mergeOut = mergeAndCount(output1.list, output2.list);    
        return {
          'count': output1.count + output2.count + mergeOut.count,
          'list': mergeOut.list
        };
      }
    };
    
    console.clear();
    var r = sortAndCount([1,3,4,2]);
    console.log('RESULT',r.list);
    

    DEMO: http://jsbin.com/UgUYocu/2/edit