Is there an indexOf in javascript to search an array with custom compare function
Asked Answered
C

8

31

I need the index of the first value in the array, that matches a custom compare function.

The very nice underscorej has a "find" function that returns the first value where a function returns true, but I would need this that returns the index instead. Is there a version of indexOf available somewhere, where I can pass a function used to comparing?

Thanks for any suggestions!

Cassiodorus answered 10/9, 2012 at 17:28 Comment(3)
I think your approach in general is wrong here. You do not want functionality to modify default functionality (overloading ===), you want your own functionality (eg; myIndexOf). The former is more disruptive and dangerous than the latter.Shortwave
lodash has it lodash.com/docs#findIndexNecrose
If your target environment supports ES2015 (or you have a transpile step, eg with Babel), you can use the native Array.prototype.findIndex().Gird
J
29

Here's the Underscore way to do it - this augments the core Underscore function with one that accepts an iterator function:

// save a reference to the core implementation
var indexOfValue = _.indexOf;

// using .mixin allows both wrapped and unwrapped calls:
// _(array).indexOf(...) and _.indexOf(array, ...)
_.mixin({

    // return the index of the first array element passing a test
    indexOf: function(array, test) {
        // delegate to standard indexOf if the test isn't a function
        if (!_.isFunction(test)) return indexOfValue(array, test);
        // otherwise, look for the index
        for (var x = 0; x < array.length; x++) {
            if (test(array[x])) return x;
        }
        // not found, return fail value
        return -1;
    }

});

_.indexOf([1,2,3], 3); // 2
_.indexOf([1,2,3], function(el) { return el > 2; } ); // 2
Jactitation answered 10/9, 2012 at 17:49 Comment(3)
Underscore is overkill. You might want to wrap the whole thing in an IIFE: you've just introduced a dependency on a global variable. Also if there's a bug in your code you've contaminated all code using _.indexOf.Amabel
@mintsauce - The OP referenced Underscore, that's why I offered an Underscore-based solution. W/r/t the global reference, this is a snippet, not a drop-in module; it's the job of the user to wrap it or otherwise set it up in a way appropriate for their application. W/r/t the bug - true, that's why I prefer to write bug-free code :).Jactitation
Note: underscore.js added a findIndex function since the original post and answer.Evers
W
15

There's a standard function in ECMAScript 2015 for Array.prototype.findIndex(). Currently it's implemented in all major browsers apart from Internet Explorer.

Here's a polyfill, courtesy of the Mozilla Developer Network:

// https://tc39.github.io/ecma262/#sec-array.prototype.findIndex
if (!Array.prototype.findIndex) {
  Object.defineProperty(Array.prototype, 'findIndex', {
    value: function(predicate) {
     // 1. Let O be ? ToObject(this value).
      if (this == null) {
        throw new TypeError('"this" is null or not defined');
      }

      var o = Object(this);

      // 2. Let len be ? ToLength(? Get(O, "length")).
      var len = o.length >>> 0;

      // 3. If IsCallable(predicate) is false, throw a TypeError exception.
      if (typeof predicate !== 'function') {
        throw new TypeError('predicate must be a function');
      }

      // 4. If thisArg was supplied, let T be thisArg; else let T be undefined.
      var thisArg = arguments[1];

      // 5. Let k be 0.
      var k = 0;

      // 6. Repeat, while k < len
      while (k < len) {
        // a. Let Pk be ! ToString(k).
        // b. Let kValue be ? Get(O, Pk).
        // c. Let testResult be ToBoolean(? Call(predicate, T, « kValue, k, O »)).
        // d. If testResult is true, return k.
        var kValue = o[k];
        if (predicate.call(thisArg, kValue, k, o)) {
          return k;
        }
        // e. Increase k by 1.
        k++;
      }

      // 7. Return -1.
      return -1;
    },
    configurable: true,
    writable: true
  });
}
Wendalyn answered 8/12, 2014 at 12:35 Comment(2)
findIndex actually made it to the EcmaScript 2015 standard in the meantime, see link provided by @Wendalyn aboveCassiodorus
Underscore has also a findIndex() function in case the browser doesn't support itCassiodorus
M
7

You could do something like this:

Array.prototype.myIndexOf = function(f)
{
    for(var i=0; i<this.length; ++i)
    {
        if( f(this[i]) )
            return i;
    }
    return -1;
};

Regarding Christian's comment: if you override a standard JavaScript method with a custom one with a different the same signature and different functionality, bad thing will likely happen. This is especially true if you're pulling in 3rd party libraries which may depend on the original, say, Array.proto.indexOf. So yeah, you probably want to call it something else.

Messieurs answered 10/9, 2012 at 17:35 Comment(3)
Thanks for the concern. I also believe in second chances. ;) Please do highlight why Array.prototype.indexOf(function) is the wrong approach, and I'll give you that upvote.Shortwave
Thanks for this one. I could use this one without adding it to Array.prototype.Cassiodorus
It's better not to add to Array.prototype at all unless you are providing a shim for a function that is in the Standard but not supported by a particular implementation.Amabel
E
3

As others have noted, easy enough to roll your own, which you can keep short and simple for your particular use case:

// Find the index of the first element in array
// meeting specified condition.
//
var findIndex = function(arr, cond) {
  var i, x;
  for (i in arr) {
    x = arr[i];
    if (cond(x)) return parseInt(i);
  }
};

var moreThanTwo = function(x) { return x > 2 }
var i = findIndex([1, 2, 3, 4], moreThanTwo)

Or if you're a CoffeeScripter:

findIndex = (arr, cond) ->
  for i, x of arr
    return parseInt(i) if cond(x)
Euphonium answered 5/7, 2014 at 16:6 Comment(0)
F
1

The javascript array method filter returns a subset of the array that return true from the function passed.

var arr= [1, 2, 3, 4, 5, 6],
first= arr.filter(function(itm){
    return itm>3;
})[0];
alert(first);

if you must support IE before #9 you can 'shim' Array.prototype.filter-

Array.prototype.filter= Array.prototype.filter || function(fun, scope){
    var T= this, A= [], i= 0, itm, L= T.length;
    if(typeof fun== 'function'){
        while(i<L){
            if(i in T){
                itm= T[i];
                if(fun.call(scope, itm, i, T)) A[A.length]= itm;
            }
            ++i;
        }
    }
    return A;
}
Fou answered 10/9, 2012 at 18:14 Comment(2)
thanks, building a new subset array might slow down performance - what do you think?Cassiodorus
I think this is the simplest, most elegant answer. Doesn't require defining any new methods, underscore, or a polyfill or anything. I'm talking strictly about simplicity, not performance.Fong
D
1

How about such find function ?

(function () {
  if (!Array.prototype._find) {
    Array.prototype._find = function (value) {
      var i = -1, j = this.length;
      if (typeof(value)=="function") 
         for(; (++i < j) && !value(this[i]););
      else
         for(; (++i < j) && !(this[i] === value););

      return i!=j ? i : -1;
    }
  }
}());
Diaster answered 27/5, 2013 at 12:41 Comment(0)
V
1

Here comes the coffeescript version of nrabinowitz's code.

# save a reference to the core implementation
indexOfValue = _.indexOf

# using .mixin allows both wrapped and unwrapped calls:
# _(array).indexOf(...) and _.indexOf(array, ...)
_.mixin ({
    # return the index of the first array element passing a test
    indexOf: (array, test) ->
        # delegate to standard indexOf if the test isn't a function
        if (!_.isFunction(test))
            return indexOfValue(array, test)
        # otherwise, look for the index
        for item, i in array
            return i if (test(item))
        # not found, return fail value
        return -1
})
Visconti answered 6/8, 2013 at 1:51 Comment(0)
C
0

using underscore I came up with something copied from their find implementation using _.any:

findIndex = function (obj, iterator, context) {
    var idx;
    _.any(obj, function (value, index, list) {
        if (iterator.call(context, value, index, list)) {
            idx = index;
            return true;
        }
    });
    return idx;
};

What do you think - do you have any better solutions?

Cassiodorus answered 10/9, 2012 at 17:40 Comment(2)
I don't know what _.any means. I guess you're using some framework? If so, be nice to us and tell us your secret ;).Shortwave
You should not use the extra function with any, it slows the method down. Just use a simple for-loopBrazell

© 2022 - 2024 — McMap. All rights reserved.