简体   繁体   中英

Can someone tell me why this JavaScript code isn't lining up an array in order?

Live code: http://jsfiddle.net/fCUZC/

//INPUT ARRAY:
var input = [28,32,21,11,8,2,14,32,64];
//VARIABLE DECLARATION. a = highest number so far, b = position of that number
entireLoop:
for (var i = 1; i<=input.length; i++)
{
    if(input[i] > input[i-1])
    {
        for(var o = i; o>=0; o--)
        {
            if(input[i-1] > input[o])
            {
                input.splice(i,0,input[o]);
                input.splice((o+1),1);
                continue entireLoop;
            }
            else if(input[o] > input[0])
            {
                input.splice(0,0,input[o]);
                input.splice((o+1),1);
                continue entireLoop;
            }

        }
    }
}
document.write(input);

I'm trying to order the array from largest to smallest, but there's a 32 stuck somewhere. I know there's the sort method, but I'm a newbie and want to try this for myself.

** edit ** First have a look at the Array's native .sort() method. It leaves the original array intact and accepts a comparison function. The latter makes .sort() pretty powerful.

var input = [28,32,21,11,8,2,14,32,64];

var low2high = function ( a , b ) {
    return a > b;
};

var high2low = function ( a , b ) {
    return a < b;
};

var resultHigh2low = input.sort( high2low ); // [ 64,32,32,28,21,14,11,8,2 ];
var resultLow2high = input.sort( low2high ); // [ 2,8,11,14,21,28,32,32,64 ];

So if we want to use bubbleSort ( link provided by TJ Crowder , see OP comments ) we can write the following:

// javascript bubbleSort implementation
var bubbleSort = function ( list , comparison ) {
    var swapped;
    var i;
    var val;

    list = [].concat( list ); // do not destroy original
    comparison = ( typeof comparison == "function" ) ? comparison : function(a,b){return a > b;}

    do {
        i = list.length;
        while ( --i ) {
           if ( i && comparison( list[ i ] , comparison[ i-1] ) ) {
                val = list[ i ];
                list[ i ] = list[ i - 1 ];
                list[ i - 1] = val;
                swapped = true;
           }
        }
    } while ( swapped );
    return list;
}

// using comparison functions from previous example.
var resultHigh2low = bubbleSort( input , high2low ); // [ 64,32,32,28,21,14,11,8,2 ];
var resultLow2high = bubbleSort( input , low2high ); // [ 2,8,11,14,21,28,32,32,64 ];

Lets walk through it step by step:

var bubbleSort = function ( list , comparison ) {
    ..code..
}

Our function accepts 2 parameters, first the array and 2nd an optional comparison function.

var swapped;
var i = list.length;
var val;

We store the list's length under variable i , and declare 2 empty variables ( swapped and val ) we're going to use later on.

list = [].concat( list ); // do not destroy original

We clone the list using [].concat( array ) and overwrite the local list variable leaving the original intact.

comparison = ( typeof comparison == "function" ) ? comparison : function(a,b){return a > b;}

We test the typeof the comparison argument, if it's a function we use that one, otherwise we fall back on our own comparison function. Our fallback comparison function will return true if a is bigger than b .

do {
    ..code..
} while ( swapped );

A do/while loop will run at least once, our swapped variable is currently undefined so it will be interpreted as falsy. If our comparison function returns true, a swap occurs and the swapped variable will be set to true, so it will loop again.

while ( --i ) {
    ..code..
}

Here I loop from the list's length downward, the -- operator is put before the i variable to ensure it is handled first before anything, i-- would go off after while evaluation causing erronous results since list[ list.length ] does not exist. I always do it this way (bad habbit perhaps), but if it confuses you, go for absolute transparancy.

if ( i && comparison( list[ i ] , comparison[ i-1] ) ) {
    ..code..
}

First we check if i has a truthy value ( 0 evaluates to falsy ) and then we run the comparison function passing list[ i ] and list[ i - 1 ] as a and b parameters. If the comparison function returns true , we perform a swap.

val = list[ i ];
list[ i ] = list[ i - 1 ];
list[ i - 1] = val;
swapped = true;

Here I perform the swap without using the .splice() method, it's just an educated guess atm., but I figure direct assignments are faster then function calls. I use the val variable as a place holder. After the swap is done, I set swapped to true so our do/while loop will continue.

return list;

Well... return the result.

I've excluded some checks, like what do we do when the list's length is 0 and whatnot. Basically when writing helper functions, we also need to deal with error handling. Like for example throwing a TypeError when the passed comparison argument is not a function, ensuring the comparison method returns a boolean value and so on.

//INPUT ARRAY:
var input = [28,32,21,11,8,2,14,32,64];
//VARIABLE DECLARATION. a = highest number so far, b = position of that number
for (var i = 1; i<input.length; i++)
{
    if(input[i] > input[i-1])
    {
        for(var o = i-1; o>=0; o--)
        {
            if(input[i] > input[o])
            {
                input.splice(i+1,0,input[o]);
                input.splice((o),1);
                i--;
            }       
        }
    }
}
document.write(input);

While it's still not great, it should work. Keep in mind I barely tested this and I'm fairly inexperienced with javascript. Your intentions weren't all bad and everyone needs to start somewhere.

The biggest issue was simply the inner conditional. I can see your logic of looping backwards from the large value you've found and pushing all smaller values to the right. Unfortunately your indexes are a little off. You also need to run this loop until it finishes normally instead of continuing, else you'll only switch one value. When this conditional is fixed the second one is no longer needed

The alternate form would have been to start from the lowest index and find the first value that is smaller than input[i] and place it there. This is potentially clearer.

I think this was a pretty good first shot and wasn't that hard to get working. Good luck!

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM