javascript знайти та видалити об’єкт у масиві на основі значення ключа


138

Я намагався декілька підходів щодо того, як знайти об’єкт у масиві, де 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

Я можу присвоїти масив id, але як я можу видалити весь об'єкт, де id == 88

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

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

12
Ця відповідь дає найбільш стисле та ідіоматичне рішення для jQuery
Брайан

1
У випадку, коли ви хочете видалити всі елементи з id = щось добре, але будьте обережні, використовуючи $ .grep, оскільки він шукає повний масив, а для довгих масивів це не ефективно. Іноді потрібно просто перевірити, чи існує елемент у масиві заданого ідентифікатора, тоді краще скористатися іншим методом ітерації;)
julianox

1
Це не видаляє цей об’єкт зі списку
Arun Sivan

1
@ArunSivan також sliceнічого не видаляє. Не впевнений, у що ви потрапляєте. Якщо у вас є певна проблема, ви можете поставити нове запитання .
Бергі

1
@Learnerdata.filter(e => !ids.includes(e.id))
Берги

151

ось рішення, якщо ви не використовуєте 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;
    }
}

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


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!");

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 хвилин життя.
Джеймс Хіббард

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" }]

1
Noooooo! Не використовуйте jQueryкарту як фільтр.
Бергі

1
Погодьтеся! Ваше рішення з grep - це правильне рішення з jQuery.
некман

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.


Чому немає анотацій? Якщо ця справа належним чином написана і у вас немає страшних помилок, вона повинна бути надзвичайно корисною.
Макс Ярі

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


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];
  });
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.