Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Random array selection without selecting twice in JavaScript

So, I am attempting to select a random entry from an array, and then make it so that particular entry will not be selected again until every entry has been selected. Basically, I don't want to see any of the same entries, until all of the entries in the array have been selected.

So if this were my array…

keywords =
[
 "ppc",
 "games",
 "advertise",
 "meta",
 "home",
 "gaming",
 "welcome"
]
var keyword = keywords[Math.floor(Math.random()*keywords.length)]
document.write(keyword);

I would not want to see an output of:

meta, advertise, home, meta, gaming, welcome, ppc, welcome

since meta was selected a second time before everything had been selected once. I would like to see something more like:

meta, advertise, gaming,ppc, welcome, home, games, advertise, ppc,

since this did not select any entry multiple times before every entry had been randomly selected.( the second loop started at the second "advertise" in case you didn't catch the differences.

But as you can see from the code that I have posted above, I do not know how to do this. I have seen examples where the entries that were randomly selected, were actually deleted from the array entirely but this is not what I want to do. I just want every entry to be selected once, and then for the process to be restarted.

Does anyone know the code for this?

like image 313
Andreas Jarbol Avatar asked Aug 20 '11 17:08

Andreas Jarbol


4 Answers

You could use the Array.sort() function to sort it randomly.

// random sort function
function shuffle(a, b)
{
   return Math.random() > 0.5 ? -1 : 1;
}

var keywords = ["ppc", "games", "advertise", "meta", "home", "gaming", "welcome"];

var randomKeywords = keywords.sort(shuffle); // new instance of a sorted randomly copy of the array

alert(randomKeywords);

update:

A better solution for shuffling is using Fisher-Yates Shuffle, as found in this answer.

function shuffle(array)
{
  var m = array.length, t, i;
  while (m > 0) 
  {
	i = Math.floor(Math.random() * m--);
	t = array[m];
	array[m] = array[i];
	array[i] = t;
  }
  return array;
}

var keywords = ["ppc", "games", "advertise", "meta", "home", "gaming", "welcome"];

shuffle(keywords); // shuffles the array

alert(keywords);
like image 98
Mark Knol Avatar answered Oct 03 '22 23:10

Mark Knol


A very simplistic way to do this would be to use splice every times you select a random element and once the array is empty re-fill it with the original values.

Example :

(function () {
    var arr = [];

    window.getRandomThing = function () {
        if (arr.length === 0) {
            refill();
        }

        return arr.splice(Math.random() * arr.length, 1)[0];
    };

    function refill () {
        arr = [1,2,3,4,5];
    }
} ());
like image 43
HoLyVieR Avatar answered Oct 03 '22 23:10

HoLyVieR


You can make a copy of the original Array, then use .splice() to grab a value at a random index, and remove it from the copy Array.

Because the copy is being reduced by one each time, you can simply do it while( copy.length ).

Example: http://jsfiddle.net/fMXTF/

var keywords = [
 "ppc",
 "games",
 "advertise",
 "meta",
 "home",
 "gaming",
 "welcome"
];

var copy = keywords.slice();

while( copy.length ) {

    var keyword = copy.splice( Math.floor(Math.random()*copy.length), 1 );
    document.write(keyword + '<br>');

}

Notice that the random number is based off copy.length, which, because of the .splice(), is reduced by 1 in each iteration. Therefore it ensures the random number is always based on the current length of the copy.

like image 38
user113716 Avatar answered Oct 03 '22 23:10

user113716


If you don't mind the array changing you could randomize the order of the elements in the array and then print the array from first element to last.

OR

You could make another array of values 1 to N (where n is the number of elements). Randomize the order of that array and then use that as an index to the array as you iterate over it from the first to last.

like image 24
Hogan Avatar answered Oct 04 '22 00:10

Hogan