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?
Stolen from user113716, but optimized it a bit.
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 :
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/
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 currentlength
of the copy.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.
Store the numbers (indices) you've seen in a hash, then when you try look at a new word you can check the hash, if you've already seen it, generate a new number. Make sure to check to see if the hash length is the same as the array length though.
This avoids altering the array.
You could use the Array.sort() function to sort it randomly.
update:
A better solution for shuffling is using Fisher-Yates Shuffle, as found in this answer.