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
当然可以,但是,我写这个问题的原因是因为我被卡住了;)有什么摘要吗?
汤姆


标题和问题文本似乎冲突……建议了两种完全不同的方法:A.从数组中删除项目与 B.创建一个新的,经过过滤的数组。
佳能

Answers:


155

我可以grep数组的id,但如何删除id == 88的整个对象

只需通过相反的谓词过滤即可:

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

12
该答案为jQuery提供了最简洁,最惯用的解决方案
Bryan

1
如果您想删除id = something的所有项目,那是可以的……但是在使用$ .grep时要小心,因为它会搜索整个数组,而对于长数组则搜索效率不高。有时,您只需要通过给定的ID检查元素是否存在于数组中,则最好使用另一种迭代方法;)
julianox 2014年

1
这不会从列表中删除该对象
Arun Sivan

1
@ArunSivan slice也不会删除任何内容。不知道你在说什么。如果您自己有特定问题,则可能要问一个新问题
Bergi '17

1
@Learnerdata.filter(e => !ids.includes(e.id))
Bergi

151

如果您不使用jQuery,这是一个解决方案:

myArray = myArray.filter(function( obj ) {
  return obj.id !== id;
});

2
这是不是做的更好findIndex(),然后splice(index, 1)父阵列上?
亚历克斯(Alex)

splice改变原始数组。使用过滤器,您可以选择。
velop

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;
    }
}

只需遍历列表,找到匹配的ID,进行拼接,然后中断即可退出循环


17
+1,但您应该提及这确实会删除仅匹配的第一项。
Bergi 2014年

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和array spread运算符的新方法:

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

假设id是唯一的,而您只需要删除一个元素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
不!请勿将jQuery地图用作过滤器。
Bergi 2014年

1
同意!使用grep的解决方案是使用jQuery的正确解决方案。
nekman 2014年

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集合过滤。


为什么没有upwotes?如果这个东西写得正确并且没有可怕的错误,那么它应该非常有用。
Max Yari

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

如果测试严格相等,请确保将对象ID强制为整数:

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];
  });
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.