Reputation: 65510
I'm using JavaScript, and would like to check whether an array exists in an array of arrays.
Here is my code, along with the return values:
var myArr = [1,3];
var prizes = [[1,3],[1,4]];
prizes.indexOf(myArr);
-1
Why?
It's the same in jQuery:
$.inArray(myArr, prizes);
-1
Why is this returning -1 when the element is present in the array?
Upvotes: 44
Views: 51820
Reputation: 783
Use js Maps, have a hash of the array as keys and actual arrays as values, if you need to iterate through all the arrays you can do map.values(). if u need to see if an array exists just calculate the hash o(n) and lookup o(1).
hash function can be as simple as concatinating all the elements with '-' , if your array size is huge, make a number from that array ([1,2] => 12) , and use mod of a big prime number, for collision chain them.
Upvotes: -1
Reputation: 106
Not a js expert, but just figured out how to do this with Array.every
and Array.some
To find the indices of matches:
let anarr = ['a',1, 2]
let arrofarrs = [['a',1, 2], [2,3,4]]
arrofarrs.map(
subarr => subarr.every(
(arr_elem, ind) => arr_elem == anarr[ind]
)
)
// output
// Array [ true, false ]
And to check true
/false
if the array contains a subarray just change map
to some
arrofarrs.some(
subarr => subarr.every(
(arr_elem, ind) => arr_elem == anarr[ind]
)
)
// output
// true
ofc that only works for a single layer of nesting, but could be made recursive ;)
Upvotes: 1
Reputation: 145
function checkArrayInArray(arr, farr){
if(JSON.stringify(arr).includes(JSON.stringify(farr))) return true;
return false;
}
Upvotes: 2
Reputation: 386550
You could iterate the array of arrays with Array#some
and then check every item of the inner array with the single array with Array#every
.
var array = [1, 3],
prizes = [[1, 3], [1, 4]],
includes = prizes.some(a => array.every((v, i) => v === a[i]));
console.log(includes);
Upvotes: 14
Reputation: 655
You can use this
var a = [ [1,2] , [3,4] ];
var b = [1,2];
a = JSON.stringify(a);
b = JSON.stringify(b);
then you can do just an indexOf() to check if it is present
var c = a.indexOf(b);
if(c != -1){
console.log('element present');
}
Upvotes: 51
Reputation: 9685
function doesArrayOfArraysContainArray (arrayOfArrays, array){
var aOA = arrayOfArrays.map(function(arr) {
return arr.slice();
});
var a = array.slice(0);
for(let i=0; i<aOA.length; i++){
if(aOA[i].sort().join(',') === a.sort().join(',')){
return true;
}
}
return false;
}
Worth noting:
aOA[i].sort().join(',') === a.sort().join(',')
is a useful way to check for arrays that contain the same values in the same order, but are references to different objects.
array.slice(0)
creates a non-referential copy of the original 2D array.
However, to create a copy of a 3D array arrayOfArrays.slice(0)
does not work; the reference chain will still be present. In order to create a non-referential copy the .map
function is necessary.
If you don't create these non-referential array copies you can really run into some tough to track down issues. This function should operate as a conditional and not effect the initial objects passed in.
Javascript is one fickle mistress.
Upvotes: 0
Reputation: 105876
Because [1,3] !== [1,3]
, since objects will only equal if they reference the same object. You need to write your own search routine:
function searchForArray(haystack, needle){
var i, j, current;
for(i = 0; i < haystack.length; ++i){
if(needle.length === haystack[i].length){
current = haystack[i];
for(j = 0; j < needle.length && needle[j] === current[j]; ++j);
if(j === needle.length)
return i;
}
}
return -1;
}
var arr = [[1,3],[1,2]];
var n = [1,3];
console.log(searchForArray(arr,n)); // 0
If both operands are objects, they're compared as objects, and the equality test is true only if both refer the same object.
Upvotes: 21
Reputation: 3184
Assuming you are only dealing with a two-dimensional array (you mention an "array of arrays" but nothing deeper than that), this non-recursive code should do what you need.
var compare_arrays = function (array_a, array_b) {
var rtn = true,
i, l;
if (array_a.length === array_b.length) {
for (i = 0, l = array_a.length; (i < l) && rtn; i += 1) {
rtn = array_a[i] === array_b[i];
}
} else {
rtn = false;
}
return rtn;
},
indexOfSimilarArray = function (arrayToFind, arrayToSearch) {
var i = arrayToSearch.length,
chk = false;
while (i && !chk) {
i -= 1;
chk = compare_arrays(arrayToFind, arrayToSearch[i]);
}
return i;
};
// Test
var myArr = [1,3];
var prizes = [[1,3],[1,4]];
indexOfSimilarArray(myArr, prizes);
JSFiddle: http://jsfiddle.net/guypursey/V7XpE/. (View the console to see the result.)
Upvotes: 0
Reputation: 3439
first define a compare function for arrays
// attach the .compare method to Array's prototype to call it on any array
Array.prototype.compare = function (array) {
// if the other array is a falsy value, return
if (!array)
return false;
// compare lengths - can save a lot of time
if (this.length != array.length)
return false;
for (var i = 0; i < this.length; i++) {
// Check if we have nested arrays
if (this[i] instanceof Array && array[i] instanceof Array) {
// recurse into the nested arrays
if (!this[i].compare(array[i]))
return false;
}
else if (this[i] != array[i]) {
// Warning - two different object instances will never be equal: {x:20} != {x:20}
return false;
}
}
return true;
}
second just find the array with
prizes.filter(function(a){ return a.compare(myArr)})
NOTE: check the browser compatibility for array.filter
Upvotes: 1
Reputation: 80011
Because javascript objects are compared by identity, not value. So if they don't reference the same object they will return false.
You need to compare recursively for this to work properly.
Upvotes: 1
Reputation: 437336
Because both these methods use reference equality when operating on objects. The array that exists and the one you are searching for might be structurally identical, but they are unique objects so they won't compare as equal.
This would give the expected result, even if it's not useful in practice:
var myArr = [1,3];
var prizes = [myArr,[1,4]];
prizes.indexOf(myArr);
To do what you wanted you will need to write code that explicitly compares the contents of arrays recursively.
Upvotes: 5