javascript encontre e remova objeto na matriz com base no valor da chave

139

Eu tenho tentado várias abordagens sobre como encontrar um objeto em uma matriz, onde ID = var e, se encontrado, remover o objeto da matriz e retornar a nova matriz de objetos.

Dados:

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

Eu sou capaz de pesquisar a matriz usando jQuery $ grep;

var id = 88;

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

Mas como posso excluir o objeto inteiro quando id == 88 e retornar dados como este:

Dados:

[
    {"id":"99","name":"Have fun boys and girls"},
    {"id":"108","name":"You are awesome!"}
]
Tom
fonte
Que tal usar a slicefunção e um pequeno forloop?
Ahmed Hamdy
1
Claro, mas, razão pela qual escrevi esta pergunta, é porque estou preso;) algum trecho?
Tom
Verifique esta publicação stackoverflow.com/questions/10827894/…
Ahmed Hamdy
O texto do título e da pergunta parece conflitar ... sugerindo duas abordagens totalmente diferentes: A. remover itens de uma matriz versus B. criar uma nova matriz filtrada.
canon

Respostas:

155

Posso grep a matriz para a identificação, mas como posso excluir o objeto inteiro onde id == 88

Basta filtrar pelo predicado oposto:

var data = $.grep(data, function(e){ 
     return e.id != id; 
});
Bergi
fonte
12
Esta resposta fornece a mais concisa e solução idiomática para jQuery
Bryan
1
No caso em que você deseja excluir todos os itens com id = algo está bom ... mas tenha cuidado ao usar $ .grep, pois ele pesquisa toda a matriz e, para matrizes longas, não é eficiente. Às vezes você só precisa verificar se o elemento existe dentro da matriz por um determinado ID, então é melhor usar outro método iteração;)
julianox
1
Isso não remove o objeto da lista
Arun Sivan
1
@ArunSivan slicetambém não remove nada. Não sei ao que você está chegando. Se você tiver um problema específico, faça uma nova pergunta .
Bergi
1
@Learnerdata.filter(e => !ids.includes(e.id))
Bergi
153

Aqui está uma solução se você não estiver usando o jquery:

myArray = myArray.filter(function( obj ) {
  return obj.id !== id;
});
Adam Boostani
fonte
2
Isso é melhor do que fazer findIndex()e depois splice(index, 1)na matriz pai?
Alex
emenda altera a matriz de origem. Com filtro você tem a escolha.
velop
13
Você pode reduzir isso para uma única linha usando: myArr = myArray.filter (obj => obj.id! == id);
precisa saber é o seguinte
2
ainda mais concisoarr = arr.filter( obj => obj.id !== id);
Omar
86

Você pode simplificar isso, e realmente não há necessidade de usar o jquery aqui.

var id = 88;

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

Basta percorrer a lista, encontrar o ID correspondente, emendar e depois interromper para sair do loop

Bryan
fonte
17
+1, mas você deve mencionar que isso exclui apenas o primeiro item correspondente.
Bergi
6
... E se você precisar excluir cada item correspondente, faça um loop na ordem inversa com i=data.length; i > 0; i--e não use break.
Jeremy Belolo 28/09
3
i = data.lengthvai quebrar qualquer data[i], deve ser algo comoi=data.length -1 ; i > -1; i--
distante
31

Há um novo método para fazer isso no ES6 / 2015 usando o findIndex e o operador de spread de matriz:

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

Você pode transformá-lo em uma função para reutilização posterior como esta:

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

Dessa forma, você pode remover itens por chaves diferentes usando um método (e se não houver objeto que atenda aos critérios, você retornará a matriz original):

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

Ou você pode colocá-lo em seu 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;
};

E use-o desta maneira:

const newData = data.remove("id", "88");
const newData2 = data.remove("name", "You are awesome!");
zorza
fonte
findIndex () é realmente ótimo! 👍
danielcraigie
9

Supondo que os IDs sejam únicos e você só precisará remover o elemento que splicedeve executar o truque:

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);
James Hibbard
fonte
Você tem os elementos em sua função de retorno de chamada para trás. Deveria ser each(data,function(idx,ele). Vou cobrá-lo mais tarde para o 30 min I desperdiçado descobrir isso :)
CSharper
Opa O mínimo que posso fazer nesse caso é atualizar minha resposta. Sinto-me muito mal pelos seus 30 minutos desperdiçados de vida.
James Hibbard
5
var items = [
  {"id":"88","name":"Lets go testing"},
  {"id":"99","name":"Have fun boys and girls"},
  {"id":"108","name":"You are awesome!"}
];

Se você estiver usando jQuery, use jQuery.grep assim:

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

Usando ES5 Array.prototype.filter :

items = items.filter(function(item) { 
  return item.id !== '88'; 
});
// items => [{ id: "99" }, { id: "108" }]
nekman
fonte
1
Nããããão! Não use o jQuerymapa como um filtro.
Bergi
1
Aceita! Sua solução com grep é a solução certa com jQuery.
9114 nekman
4

Talvez você esteja procurando $.grep()função:

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
});
Rafael Garcia
fonte
3

sifté um poderoso filtro de coleção para operações como essa e muito mais avançadas. Funciona do lado do cliente no navegador ou do servidor no 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);

Ele suporta filtros como $in, $nin, $exists, $gte, $gt, $lte, $lt, $eq, $ne, $mod, $all, $and, $or, $nor, $not, $size, $type, e $regex, e se esforça para ser API compatível com filtragem coleção MongoDB.

Redsandro
fonte
Por que não fazer upwotes? Se essa coisa for escrita corretamente e não tiver erros terríveis, deve ser extremamente útil.
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;
}

Isso deve funcionar, jsfiddle

casraf
fonte
0

Certifique-se de coagir o ID do objeto a um número inteiro se você testar a igualdade estrita:

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

Demo

Andy
fonte
0

Se você estiver usando sublinhado js, ​​é fácil remover o objeto com base na chave. http://underscorejs.org . Exemplo:

  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];
  });
Mohammed Safeer
fonte