2013-07-30 12 views
6

considerare i seguenti matrici:funzione di chiamata JS con tutte le possibili argomentazioni permutata

var array1 = [true, false]; 
var array2 = [1, 2]; 
var array3 = ["a", "b", "c"]; 

Voglio chiamare la mia funzione myFunc(arg1, arg2, arg3) con tutte le combinazioni di argomenti. Ma voglio evitare di "foreach" l'inferno.

E 'possibile funzione di scrittura che mi permette che, in modo che io possa chiamare alcuni, come:

cartesianCall(array1, array2, array3, myFunc); 

idealmente con numero variabile di array (argomenti myFunc)?

EDIT: così la funzione sarebbe stata chiamata:

myFunc(true, 1, "a"); 
myFunc(true, 1, "b"); 
myFunc(true, 1, "c"); 
myFunc(true, 2, "a"); 
myFunc(true, 2, "b"); 
myFunc(true, 2, "c"); 
myFunc(false, 1, "a"); 
myFunc(false, 1, "b"); 
myFunc(false, 1, "c"); 
myFunc(false, 2, "a"); 
myFunc(false, 2, "b"); 
myFunc(false, 2, "c"); 
+0

vuoi fondere il tuo array in uno? –

+0

voglio fare più operazioni in "myFunc" con argomenti .. –

+0

E vuoi evitare nested 'for' loops? – tymeJV

risposta

3

http://jsfiddle.net/trevordixon/zEqKy/

function cartesianCall(func, args) { 
    var combos = allCombos.apply(this, args); 

    for (var i = 0; i < combos.length; i++) { 
     func.apply(null, combos[i]); 
    } 
} 

function allCombos(first) { 
    var isArray = toString.call(first) === "[object Array]"; 
    if (!isArray) first = [first]; // Convert non-array to an array with the value 
            // as the only element 
    else if (first.length === 0) first = [undefined]; // Convert empty array to an 
                 // array with undefined as 
                 // the only element 

    if (arguments.length === 1) return first; // base case for recursion 

    var result = [], 
     rest = allCombos.apply(this, Array.prototype.slice.call(arguments, 1)); 

    for (var i = 0; i < first.length; i++) { 
     for (var j = 0; j < rest.length; j++) { 
      result.push([first[i]].concat(rest[j])); 
     } 
    } 

    return result; 
} 

Quindi utilizzare in questo modo :

function printArgs() { console.log('Called with arguments:', arguments); } 

cartesianCall(printArgs, [ 
    [true, false], 
    undefined, 
    [1, 2], 
    [], 
    'a string', 
    ['a', 'b', 'c'] 
]); 

stampe:

Called with arguments: [true, undefined, 1, undefined, "a string", "a"] 
Called with arguments: [true, undefined, 1, undefined, "a string", "b"] 
Called with arguments: [true, undefined, 1, undefined, "a string", "c"] 
Called with arguments: [true, undefined, 2, undefined, "a string", "a"] 
Called with arguments: [true, undefined, 2, undefined, "a string", "b"] 
Called with arguments: [true, undefined, 2, undefined, "a string", "c"] 
Called with arguments: [false, undefined, 1, undefined, "a string", "a"] 
Called with arguments: [false, undefined, 1, undefined, "a string", "b"] 
Called with arguments: [false, undefined, 1, undefined, "a string", "c"] 
Called with arguments: [false, undefined, 2, undefined, "a string", "a"] 
Called with arguments: [false, undefined, 2, undefined, "a string", "b"] 
Called with arguments: [false, undefined, 2, undefined, "a string", "c"] 

notare che le matrici vuote vengono trattate come undefined.

+0

Quando una matrice è una lista vuota, non permuta, dovrebbe essere permutabile con il valore null – zsong

+0

Buon punto. Penso che la modifica si occupi di quel caso limite. –

+0

Penso che quando uno degli array è vuoto, la funzione non si chiami –

5

Declare si funzione senza parametri e utilizzare arguments parola chiave:

function cartesianCall() { 
    for (var i = 0; i < arguments.length; i++) { 
    // do something with arguments[i] 
    } 
} 
+0

Questo è il modo migliore per fare le cose, basta fare attenzione a non definire alcuna funzione di aiuto all'interno del ciclo. –

+2

Questo in realtà non risponde alla domanda. Avrà ancora bisogno di fare un sacco di brutti loop in là. –

+0

@TrevorDixon La mia risposta è stata pubblicata prima che la domanda OP fosse aggiornata con ulteriori informazioni. Quindi questa è la risposta alla domanda: * È possibile la funzione di scrittura che me lo consente, quindi posso chiamarla un po 'come: cartesianCall (array1, array2, array3, myFunc); * –

Problemi correlati