次のようなJavaScriptの配列があるとします
[[1,2,3],[4,5,6],[7,8,9],[10,11,12],[13,14,15],[16,17,18]]
サブ配列の要素をランダムにシャッフルして、このようなものを取得できますか
[[16,4,10],[8,3,9],[1,14,18],[2,5,7],[6,17,11],[12,13,15]]
編集:すべてのサブアレイは同じ長さです。また、新しい配列には、古い配列と同じ長さのサブ配列が含まれます。
次のようなJavaScriptの配列があるとします
[[1,2,3],[4,5,6],[7,8,9],[10,11,12],[13,14,15],[16,17,18]]
サブ配列の要素をランダムにシャッフルして、このようなものを取得できますか
[[16,4,10],[8,3,9],[1,14,18],[2,5,7],[6,17,11],[12,13,15]]
編集:すべてのサブアレイは同じ長さです。また、新しい配列には、古い配列と同じ長さのサブ配列が含まれます。
これにはアンダースコアを使用します: http://underscorejs.org/
v = [[1,2,3],[4,5,6],[7,8,9],[10,11,12],[13,14,15],[16,17,18]]
v2 = _.shuffle(_.flatten(v))
v3 = _.groupBy(v2, function(item, i) {
return i % v.length;
})
これはその場でシャッフルし、新しい配列を返しません:
function shuffleMatrix (m) {
if (m.length == 0 || m[0].length == 0) {
// no rows or columns, just return it
return m;
}
function swap(i1, j1, i2, j2) {
if (i1 != i2 || j1 != j2) {
var temp = m[i1][j1];
m[i1][j1] = m[i2][j2];
m[i2][j2] = temp;
}
var rows = m.length;
var cols = m[0].length;
for (i = 0; i < rows; i++) {
for (j = 0; j < cols; j++) {
var new_i = Math.random()*rows;
var new_j = Math.random()*cols;
swap(i, j, new_i, new_j);
}
}
}
function arrayShuffle(){
var tmp, rand;
for(var i =0; i < this.length; i++){
rand = Math.floor(Math.random() * this.length);
tmp = this[i];
this[i] = this[rand];
this[rand] =tmp;
}
}
Array.prototype.shuffle =arrayShuffle;
その後
for(var i in arrays){
arrays[i].shuffle()
}
トップレベルの配列内に配列を配置するのと同じことです。
//編集:読み間違い:)
function shuffleAnyNumber (arrays) {
var numbers = new Array(); //collection of all numbers
for(var i in this){
for(var j in arrays[i]){
numbers.push( arrays[i][j]); //collect numbers out of the given arrays
}
}
numbers.shuffle(); //Method shown above
var output = new Array();
var tempArray= new Array();
//putting it together
for(var i in numbers){
if(tempArray.length == 3){
output.push(tempArray);
tempArray = new Array();
} else {
tempArray.push(numbers[i]);
}
}
return output;
}
そのように機能すると思います。
配列をフラット化し、シャッフルしてから、再度分割することができます。
var flat = [].concat.apply([], myArray);
arrayShuffle(flat);
var newArray = [],
sublen = myArray[0].length;
for (var i=0; i<flat.length; i+= sublen)
newArray.push(flat.slice(i, i+sublen));
または、サブ配列のアイテムを使用するように既存のシャッフル アルゴリズムの 1 つを変更します。たとえば、Fisher-Yates-shuffle:
function shuffle2dArray(array) {
if (!array.length) return array;
var sublen = array[0].length,
len = array.length * sublen;
for (var i = len - 1; i > 0; i--) {
var j = Math.floor(Math.random() * (i + 1));
var ix = ~~(i/sublen),
iy = i % sublen,
jx = ~~(j/sublen),
jy = j % sublen;
var temp = array[ix][iy];
array[ix][iy] = array[jx][jy];
array[jx][jy] = temp;
}
return array;
}
私はIlanBerciの答えを100%支持します。ただし、アンダースコアを使用して、より機能的なスタイルで記述できます。
var v = [[1,2,3],[4,5,6],[7,8,9],[10,11,12],[13,14,15],[16,17,18]];
return _(v).chain().flatten().shuffle().groupBy(function(item, i) {
return i % v.length;
}).values().value();
アンダースコアを付けてください!!