JavaScript найти и удалить объект в массиве на основе значения ключа

139

Я пробовал несколько подходов о том, как найти объект в массиве, где ID = var, и, если он найден, удалить объект из массива и вернуть новый массив объектов.

Данные:

[
    {"id":"88","name":"Lets go testing"},
    {"id":"99","name":"Have fun boys and girls"},
    {"id":"108","name":"You are awesome!"}
]

Я могу искать в массиве, используя jQuery $ grep;

var id = 88;

var result = $.grep(data, function(e){ 
     return e.id == id; 
});

Но как я могу удалить весь объект, когда id == 88, и вернуть данные так:

Данные:

[
    {"id":"99","name":"Have fun boys and girls"},
    {"id":"108","name":"You are awesome!"}
]
Том
источник
Как насчет использования sliceфункции и небольшого forцикла?
Ахмед Хамди,
1
Конечно, но причина, по которой я написал этот вопрос, в том, что я застрял;) какие-нибудь фрагменты?
Том
Проверьте это сообщение stackoverflow.com/questions/10827894/…
Ахмед Хэмди
Текст заголовка и вопроса кажется противоречивым ... предлагая два совершенно разных подхода: A. удалить элементы из массива по сравнению с B. создать новый, отфильтрованный массив.
канон

Ответы:

155

Я могу grep массив для идентификатора, но как я могу удалить весь объект, где id == 88

Просто отфильтруйте по противоположному предикату:

var data = $.grep(data, function(e){ 
     return e.id != id; 
});
Берги
источник
12
Этот ответ дает наиболее краткое и идиоматическое решение для jQuery
Брайан,
1
В том случае, если вы хотите удалить все элементы с id = что-то, это хорошо ... но будьте осторожны при использовании $ .grep, поскольку он ищет полный массив и для длинных массивов это неэффективно. Иногда вам просто нужно проверить, существует ли элемент внутри массива по заданному идентификатору, тогда лучше использовать другой метод итерации;)
julianox
1
Это не удаляет этот объект из списка
Арун Сиван
1
@ArunSivan тоже sliceничего не удаляет. Не уверен, к чему ты клонишь. Если у вас есть конкретная проблема, вы можете задать новый вопрос .
Берги
1
@Learnerdata.filter(e => !ids.includes(e.id))
Берги
153

Вот решение, если вы не используете jquery:

myArray = myArray.filter(function( obj ) {
  return obj.id !== id;
});
Адам Бустани
источник
2
Это лучше чем делать findIndex()и потом splice(index, 1)на родительском массиве?
Алекс
сплайс мутирует исходный массив. С фильтром у вас есть выбор.
ворачивается
13
Вы можете уменьшить это до одной строки, используя: myArr = myArray.filter (obj => obj.id! == id);
DBrown
2
еще лаконичнееarr = arr.filter( obj => obj.id !== id);
Омар
86

Вы можете упростить это, и здесь нет необходимости использовать jquery.

var id = 88;

for(var i = 0; i < data.length; i++) {
    if(data[i].id == id) {
        data.splice(i, 1);
        break;
    }
}

Просто переберите список, найдите соответствующий идентификатор, сплайс, а затем прервитесь, чтобы выйти из цикла

Bryan
источник
17
+1, но вы должны отметить, что это удаляет только первый элемент, который соответствует.
Берги
6
... А если вам нужно удалить все подходящие элементы, i=data.length; i > 0; i--выполните цикл в обратном порядке с помощью и не используйте break.
Джереми Белоло
3
i = data.lengthсломается любой data[i], должно быть что-то вродеi=data.length -1 ; i > -1; i--
distante
31

В ES6 / 2015 для этого есть новый метод, использующий findIndex и оператор распространения массива:

const index = data.findIndex(obj => obj.id === id);
const newData = [
    ...data.slice(0, index),
    ...data.slice(index + 1)
]

Вы можете превратить его в функцию для последующего повторного использования следующим образом:

function remove(array, key, value) {
    const index = array.findIndex(obj => obj[key] === value);
    return index >= 0 ? [
        ...array.slice(0, index),
        ...array.slice(index + 1)
    ] : array;
}

Таким образом, вы можете удалить элементы по разным ключам, используя один метод (и если нет объекта, который соответствует критериям, вы получите исходный массив):

const newData = remove(data, "id", "88");
const newData2 = remove(data, "name", "You are awesome!");

Или вы можете поместить его в свой Array.prototype:

Array.prototype.remove = function (key, value) {
    const index = this.findIndex(obj => obj[key] === value);
    return index >= 0 ? [
        ...this.slice(0, index),
        ...this.slice(index + 1)
    ] : this;
};

И используйте это так:

const newData = data.remove("id", "88");
const newData2 = data.remove("name", "You are awesome!");
Zorza
источник
findIndex () действительно хорош! 👍
danielcraigie
9

Предполагая, что идентификаторы уникальны, и вам нужно будет только удалить один элемент spliceдолжен сделать трюк:

var data = [
  {"id":"88","name":"Lets go testing"},
  {"id":"99","name":"Have fun boys and girls"},
  {"id":"108","name":"You are awesome!"}
],
id = 88;

console.table(data);

$.each(data, function(i, el){
  if (this.id == id){
    data.splice(i, 1);
  }
});

console.table(data);
Джеймс Хиббард
источник
У вас есть элементы в вашей функции обратного вызова. Так и должно быть each(data,function(idx,ele). Я выставлю счет тебе позже за 30 минут, которые я потратил впустую, выясняя это :)
CSharper
К сожалению. Самое меньшее, что я могу сделать в этом случае, это обновить мой ответ. Мне очень жаль, что ты потратил 30 минут жизни.
Джеймс Хиббард
5
var items = [
  {"id":"88","name":"Lets go testing"},
  {"id":"99","name":"Have fun boys and girls"},
  {"id":"108","name":"You are awesome!"}
];

Если вы используете jQuery, используйте jQuery.grep следующим образом:

items = $.grep(items, function(item) { 
  return item.id !== '88';
});
// items => [{ id: "99" }, { id: "108" }]

Используя ES5 Array.prototype.filter :

items = items.filter(function(item) { 
  return item.id !== '88'; 
});
// items => [{ id: "99" }, { id: "108" }]
nekman
источник
1
Noooooo! Не используйте jQueryкарту в качестве фильтра.
Берги
1
Согласен! Ваше решение с помощью grep - это правильное решение с помощью jQuery.
nekman
4

Может быть, вы ищете $.grep()функцию:

arr = [
  {"id":"88","name":"Lets go testing"},
  {"id":"99","name":"Have fun boys and girls"},
  {"id":"108","name":"You are awesome!"}
];

id = 88;
arr = $.grep(arr, function(data, index) {
   return data.id != id
});
Рафаэль Гарсия
источник
3

siftявляется мощным фильтром коллекции для подобных операций и для более сложных. Работает на стороне клиента в браузере или на стороне сервера в node.js.

var collection = [
    {"id":"88","name":"Lets go testing"},
    {"id":"99","name":"Have fun boys and girls"},
    {"id":"108","name":"You are awesome!"}
];
var sifted = sift({id: {$not: 88}}, collection);

Он поддерживает фильтры , такие как $in, $nin, $exists, $gte, $gt, $lte, $lt, $eq, $ne, $mod, $all, $and, $or, $nor, $not, $size, $type, и $regex, и стремится быть API совместим с фильтрацией сбора MongoDB.

Redsandro
источник
Почему нет upwotes? Если эта вещь написана правильно и не содержит ужасных ошибок, она должна быть чрезвычайно полезной.
Макс Яри
2
Array.prototype.removeAt = function(id) {
    for (var item in this) {
        if (this[item].id == id) {
            this.splice(item, 1);
            return true;
        }
    }
    return false;
}

Это должно сработать , jsfiddle

casraf
источник
0

Убедитесь, что вы приводите идентификатор объекта к целому числу, если вы проверяете на строгое равенство:

var result = $.grep(data, function(e, i) { 
  return +e.id !== id;
});

демонстрация

Энди
источник
0

Если вы используете подчеркивание js, легко удалить объект по ключу. http://underscorejs.org . Пример:

  var temp1=[{id:1,name:"safeer"},  //temp array
             {id:2,name:"jon"},
             {id:3,name:"James"},
             {id:4,name:"deepak"},
             {id:5,name:"ajmal"}];

  var id = _.pluck(temp1,'id'); //get id array from temp1
  var ids=[2,5,10];             //ids to be removed
  var bool_ids=[];
  _.each(ids,function(val){
     bool_ids[val]=true;
  });
  _.filter(temp1,function(val){
     return !bool_ids[val.id];
  });
Мохаммед Сейфер
источник