javascript znajduje i usuwa obiekt w tablicy na podstawie wartości klucza


138

Próbowałem kilku podejść, jak znaleźć obiekt w tablicy, gdzie ID = var, a jeśli zostanie znaleziony, usuń obiekt z tablicy i zwróć nową tablicę obiektów.

Dane:

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

Jestem w stanie przeszukać tablicę przy użyciu jQuery $ grep;

var id = 88;

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

Ale jak mogę usunąć cały obiekt, gdy id == 88 i zwrócić takie dane:

Dane:

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

A co z użyciem slicefunkcji i małej forpętli?
Ahmed Hamdy

1
Jasne, ale powodem, dla którego napisałem to pytanie, jest to, że utknąłem;) jakieś urywki?
Tom


Tytuł i tekst pytania wydają się ze sobą sprzeczne ... co sugeruje dwa całkowicie różne podejścia: A. usuwanie elementów z tablicy w porównaniu z B. tworzenie nowej, przefiltrowanej tablicy.
kanon

Odpowiedzi:


155

Mogę grepować tablicę dla id, ale jak mogę usunąć cały obiekt, gdzie id == 88

Po prostu przefiltruj według przeciwnego predykatu:

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

12
Ta odpowiedź zapewnia najbardziej zwięzłe i idiomatyczne rozwiązanie dla jQuery
Bryan,

1
W przypadku, gdy chcesz usunąć wszystkie elementy z id = coś jest w porządku ... ale zachowaj ostrożność podczas używania $ .grep, ponieważ przeszukuje pełną tablicę, a dla długich tablic nie jest wydajne. Czasami wystarczy sprawdzić, czy element istnieje wewnątrz tablicy o podanym identyfikatorze, wtedy lepiej użyć innej metody iteracji;)
julianox

1
To nie usuwa tego obiektu z listy
Arun Sivan

1
@ArunSivan slicerównież niczego nie usuwa. Nie jestem pewien, do czego zmierzasz. Jeśli sam masz konkretny problem, możesz zadać nowe pytanie .
Bergi

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

151

tutaj jest rozwiązanie, jeśli nie używasz jquery:

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

2
Czy to lepsze niż zrobienie, findIndex()a następnie splice(index, 1)na tablicy nadrzędnej?
Alex

splice mutuje pierwotną tablicę. Z filtrem masz wybór.
velop

13
Możesz zredukować to do pojedynczego wiersza, używając: myArr = myArray.filter (obj => obj.id! == id);
DBrown

2
jeszcze bardziej zwięźlearr = arr.filter( obj => obj.id !== id);
Omar

86

Możesz to uprościć i naprawdę nie ma potrzeby używania tutaj jquery.

var id = 88;

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

Po prostu przejrzyj listę, znajdź pasujący identyfikator, połącz, a następnie przerwij, aby wyjść z pętli


17
+1, ale powinieneś wspomnieć, że powoduje to usunięcie tylko pierwszego pasującego elementu.
Bergi

6
... A jeśli chcesz usunąć wszystkie dopasowane elementy, i=data.length; i > 0; i--wykonaj pętlę w odwrotnej kolejności z i nie używaj break.
Jeremy Belolo

3
i = data.lengthzepsuje jakiekolwiek data[i], powinno być coś takiegoi=data.length -1 ; i > -1; i--
distante

31

W ES6 / 2015 dostępna jest nowa metoda za pomocą funkcji findIndex i operatora rozproszenia tablicy:

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

Możesz przekształcić go w funkcję do późniejszego ponownego wykorzystania w następujący sposób:

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

W ten sposób możesz usunąć elementy za pomocą różnych kluczy przy użyciu jednej metody (i jeśli nie ma obiektu spełniającego kryteria, zwrócona zostanie oryginalna tablica):

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

Lub możesz umieścić go na swoim 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;
};

I użyj tego w ten sposób:

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

findIndex () jest naprawdę świetny! 👍
danielcraigie

9

Zakładając, że identyfikatory są unikalne i wystarczy usunąć tylko jeden element 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);

Masz elementy w swojej funkcji zwrotnej wstecz. Tak powinno być each(data,function(idx,ele). Później
wystawię

Ups. W takim przypadku przynajmniej mogę zaktualizować moją odpowiedź. Bardzo mi przykro z powodu zmarnowanych 30 minut życia.
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!"}
];

Jeśli używasz jQuery, użyj jQuery.grep w następujący sposób:

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

Korzystanie z ES5 Array.prototype.filter :

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

1
Nieee! Nie używaj jQuerymapy jako filtru.
Bergi

1
Zgodzić się! Twoje rozwiązanie z grep jest właściwym rozwiązaniem z jQuery.
nekman,

4

Może szukasz $.grep()funkcji:

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

siftto potężny filtr kolekcji dla operacji takich jak ta i znacznie bardziej zaawansowanych. Działa po stronie klienta w przeglądarce lub po stronie serwera w 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);

Obsługuje filtry jak $in, $nin, $exists, $gte, $gt, $lte, $lt, $eq, $ne,$mod , $all, $and, $or, $nor, $not, $size, $type, i $regex, i stara się być zgodny z API filtrowania kolekcji MongoDB.


Dlaczego nie ma upwotes? Jeśli ta rzecz jest poprawnie napisana i nie ma żadnych strasznych błędów, powinna być bardzo przydatna.
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;
}

To powinno załatwić sprawę , jsfiddle


0

Upewnij się, że wymuszasz identyfikator obiektu na liczbę całkowitą, jeśli testujesz pod kątem ścisłej równości:

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

Próbny


0

Jeśli używasz podkreślenia js, łatwo jest usunąć obiekt na podstawie klucza. http://underscorejs.org . Przykład:

  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];
  });
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.