JavaScript trova e rimuove l'oggetto nell'array in base al valore chiave


139

Ho provato diversi approcci su come trovare un oggetto in un array, dove ID = var e, se trovato, rimuovere l'oggetto dall'array e restituire il nuovo array di oggetti.

Dati:

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

Sono in grado di cercare nell'array usando jQuery $ grep;

var id = 88;

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

Ma come posso eliminare l'intero oggetto quando id == 88 e restituire dati come questo:

Dati:

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

Che ne dici di usare la slicefunzione e un piccolo forloop?
Ahmed Hamdy,

1
Certo, ma, motivo per cui ho scritto questa domanda, è perché sono bloccato;) qualche frammento?
Tom,


Il titolo e il testo della domanda sembrano in conflitto ... suggerendo due approcci completamente diversi: A. rimuovere gli elementi da una matrice contro B. creare una nuova matrice filtrata.
canone il

Risposte:


155

Posso grep l'array per l'id, ma come posso eliminare l'intero oggetto dove id == 88

Filtra semplicemente per il predicato opposto:

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

12
Questa risposta fornisce la soluzione più concisa e idiomatica per jQuery
Bryan,

1
Nel tuo caso in cui desideri eliminare tutti gli elementi con id = qualcosa va bene ... ma fai attenzione quando usi $ .grep mentre cerca l'array completo e per lunghi array non è efficiente. A volte devi solo verificare se l'elemento esiste all'interno dell'array con un determinato ID, quindi è meglio usare un altro metodo di iterazione;)
julianox

1
Questo non rimuove quell'oggetto dall'elenco
Arun Sivan,

1
@ArunSivan slicenon rimuove nulla. Non sono sicuro di cosa stai arrivando. Se hai un problema specifico, potresti voler fare una nuova domanda .
Bergi,

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

153

ecco una soluzione se non stai usando jquery:

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

2
È meglio che fare findIndex()e quindi splice(index, 1)sull'array principale?
Alex

splice muta l'array di origine. Con il filtro hai la scelta.
velop,

13
Puoi ridurlo a una sola riga usando: myArr = myArray.filter (obj => obj.id! == id);
DBrown,

2
ancora più concisoarr = arr.filter( obj => obj.id !== id);
Omar

86

Puoi semplificarlo e non c'è davvero bisogno di usare jquery qui.

var id = 88;

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

Basta scorrere l'elenco, trovare l'id corrispondente, eseguire lo splicing e quindi interrompere per uscire dal ciclo


17
+1, ma dovresti dire che questo elimina solo il primo elemento che corrisponde.
Bergi,

6
... E se devi eliminare ogni elemento corrispondente, esegui il ciclo in ordine inverso con i=data.length; i > 0; i--e non utilizzare break.
Jeremy Belolo,

3
i = data.lengthromperà qualsiasi data[i], dovrebbe essere qualcosa di similei=data.length -1 ; i > -1; i--
distante

31

Esiste un nuovo metodo per farlo in ES6 / 2015 usando findIndex e l'operatore di diffusione dell'array:

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

Puoi trasformarlo in una funzione per un successivo riutilizzo in questo modo:

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

In questo modo puoi rimuovere elementi con chiavi diverse usando un metodo (e se non c'è nessun oggetto che soddisfi i criteri, otterrai l'array originale restituito):

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

Oppure puoi metterlo sul tuo 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 usalo in questo modo:

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

findIndex () è davvero fantastico! 👍
danielcraigie,

9

Supponendo che gli ID siano univoci e dovrai rimuovere solo un elemento, splicedovresti fare il trucco:

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

Hai gli elementi nella tua funzione di callback al contrario. Dovrebbe essere each(data,function(idx,ele). Ti addebiterò più tardi per i 30 minuti che ho sprecato a capire :)
CSharper

Ops. Il minimo che posso fare in quel caso è aggiornare la mia risposta. Mi sento davvero male per i tuoi 30 minuti di vita sprecati.
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 si utilizza jQuery, utilizzare jQuery.grep in questo modo:

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

Utilizzando ES5 Array.prototype.filter :

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

1
Noooooo! Non utilizzare la jQuerymappa come filtro.
Bergi,

1
Essere d'accordo! La tua soluzione con grep è la soluzione giusta con jQuery.
Nekman,

4

Forse stai cercando la $.grep()funzione:

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è un potente filtro di raccolta per operazioni come questa e molto più avanzate. Funziona sul lato client nel browser o lato server in 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);

Esso supporta filtri come $in, $nin, $exists, $gte, $gt, $lte, $lt, $eq, $ne, $mod, $all, $and, $or, $nor, $not, $size, $type, e $regex, e si sforza di essere API-compatibile con il filtro di raccolta MongoDB.


Perché nessun voto? Se questa cosa è scritta correttamente e non ha bug terribili, dovrebbe essere estremamente utile.
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;
}

Questo dovrebbe fare il trucco, jsfiddle


0

Assicurati di forzare l'id oggetto su un numero intero se verifichi un'uguaglianza rigorosa:

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

dimostrazione


0

Se si utilizza il carattere di sottolineatura js, è facile rimuovere l'oggetto in base alla chiave. http://underscorejs.org . Esempio:

  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];
  });
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.