Jak mogę usunąć element z listy?

275

Mam listę i chcę usunąć z niej pojedynczy element. W jaki sposób mogę to zrobić?

Próbowałem sprawdzić, jakie są oczywiste nazwy tej funkcji w podręczniku użytkownika i nie znalazłem nic odpowiedniego.

David Locke
źródło
Zależy, czy chcesz go usunąć według wartości, np. „Wartości 5”, czy według indeksu / indeksów, „elementu o indeksie 5” czy „w indeksach c (5: 6,10)? Jeśli chcesz usunąć według wartości i tam są duplikaty, to czy chcesz usunąć tylko duplikaty, pierwsze lub ostatnie wystąpienie, czy wszystkie? Czy to gwarantuje, że lista zawiera Twój element / indeks? Czy musimy obsługiwać przypadek, w którym lista jest pusta? Czy musimy upewnić się, że NA został zaliczony (/ wykluczony)? Czy lista jest płaska czy może być zagnieżdżona? Ile głębokości później?
smci
2
setdiff (myList, elementToRemove)
JStrahl

Odpowiedzi:

218

W ogóle nie znam R., ale doprowadziło mnie tutaj trochę kreatywnego googlingu: http://tolstoy.newcastle.edu.au/R/help/05/04/1919.html

Kluczowy cytat stamtąd:

Nie znajduję wyraźnej dokumentacji dla R na temat usuwania elementów z list, ale próby i błędy mówią mi

myList [[5]] <- NULL

usunie piąty element, a następnie „zamknie” dziurę spowodowaną usunięciem tego elementu. To wyczerpuje wartości indeksu, więc muszę uważać na upuszczanie elementów. Muszę pracować od końca listy do przodu.

Odpowiedź na to stanowisko później w wątku stanów:

Aby usunąć element listy, zobacz R FAQ 7.1

Oraz odpowiedniej części R FAQ mówi:

... Nie ustawiaj x [i] lub x [[i]] na NULL, ponieważ spowoduje to usunięcie odpowiedniego komponentu z listy.

Co wydaje się mówić (nieco do tyłu), jak usunąć element.

Nadzieja, która pomaga, a przynajmniej prowadzi cię we właściwym kierunku.

Chad Birch
źródło
5
Dzięki, mylist [i] <- NULL jest właśnie tym sposobem.
David Locke
37
To nie działało dla mnie. Dostaję:Error in list[length(list)] <- NULL : replacement has length zero
wfbarksdale
3
W poście @Aleksandr Levchuck pokazałem, że faktycznie mam do czynienia z wektorem i muszę stworzyć nowy obiekt
wfbarksdale,
209

Jeśli nie chcesz modyfikować listy w miejscu (np. Do przekazywania listy z elementem usuniętym do funkcji), możesz użyć indeksowania: ujemne wskaźniki oznaczają „nie uwzględniaj tego elementu”.

x <- list("a", "b", "c", "d", "e"); # example list

x[-2];       # without 2nd element

x[-c(2, 3)]; # without 2nd and 3rd

Przydatne są również wektory indeksu logicznego:

x[x != "b"]; # without elements that are "b"

Działa to również z ramkami danych:

df <- data.frame(number = 1:5, name = letters[1:5])

df[df$name != "b", ];     # rows without "b"

df[df$number %% 2 == 1, ] # rows with odd numbers only
Florian Jenn
źródło
4
Twój indeks logiczny działa tylko wtedy, gdy masz ten pojedynczy element „b” w elemencie listy. Nie można usunąć, powiedzmy w x$bten sposób, ani nie można usunąć „b” z elementu listy x[[2]] = c("b","k") .
Carl Witthoft
Odnośnie pojedynczych lub wielu przedmiotów: możesz użyć %in%do testowania wielu przedmiotów. Nie jestem pewien, co rozumiesz przez „nie można usunąć x $ b” - czy masz na myśli usunięcie całej kolumny b?
Florian Jenn,
30

Oto jak usunąć ostatni element listy w R:

x <- list("a", "b", "c", "d", "e")
x[length(x)] <- NULL

Jeśli x może być wektorem, musisz utworzyć nowy obiekt:

x <- c("a", "b", "c", "d", "e")
x <- x[-length(x)]
  • Praca dla list i wektorów
Aleksandr Levchuk
źródło
@krlmlr: wręcz przeciwnie, to rozwiązanie jest bardziej ogólne niż odpowiedź Floriana, ponieważ jest polimorficzne w typie kolekcji.
Dan Barowy
@DanBarowy: Myliłem się: wydaje się, że jest to synteza odpowiedzi Chada (zaakceptowanej) i odpowiedzi Floriana ... Dobre krótkie streszczenie.
krlmlr
19

Usuwanie elementów Null z listy w jednym wierszu:

x=x[-(which(sapply(x,is.null),arr.ind=TRUE))]

Twoje zdrowie

Sukhi
źródło
2
Ten kod pęka, gdy xjest pusta lista. Zamiast tego użyj compactz plyrtego zadania.
Richie Cotton
Również, jeśli na liście nie ma wartości null, -(which(sapply(x,is.null),arr.ind=TRUE))zwraca, named integer(0)co spowoduje całkowite usunięcie tego wiersza.
user3055034
17

Chciałbym dodać, że jeśli jest to nazwana lista, której możesz po prostu użyć within.

l <- list(a = 1, b = 2)    
> within(l, rm(a))
$b
[1] 2

Możesz więc zastąpić oryginalną listę

l <- within(l, rm(a)) 

aby usunąć element o nazwie az listy l.

Kim
źródło
1
Aby zrobić wielewithin(l, rm(a, b))
Vlad
16

Jeśli masz nazwaną listę i chcesz usunąć określony element, możesz spróbować:

lst <- list(a = 1:4, b = 4:8, c = 8:10)

if("b" %in% names(lst)) lst <- lst[ - which(names(lst) == "b")]

Będzie to zrobić listę lstz elementami a, b, c. Druga linia usuwa elementb po sprawdzeniu, że istnieje (aby uniknąć wspomnianego problemu @hjv).

albo lepiej:

lst$b <- NULL

W ten sposób nie jest problemem próba usunięcia nieistniejącego elementu (np. lst$g <- NULL)

alko989
źródło
10

Istnieje pakiet rlist ( http://cran.r-project.org/web/packages/rlist/index.html ) do obsługi różnego rodzaju operacji na listach.

Przykład ( http://cran.r-project.org/web/packages/rlist/vignettes/Filtering.html ):

library(rlist)
devs <- 
  list(
    p1=list(name="Ken",age=24,
      interest=c("reading","music","movies"),
      lang=list(r=2,csharp=4,python=3)),
    p2=list(name="James",age=25,
      interest=c("sports","music"),
      lang=list(r=3,java=2,cpp=5)),
    p3=list(name="Penny",age=24,
      interest=c("movies","reading"),
      lang=list(r=1,cpp=4,python=2)))

list.remove(devs, c("p1","p2"))

Prowadzi do:

# $p3
# $p3$name
# [1] "Penny"
# 
# $p3$age
# [1] 24
# 
# $p3$interest
# [1] "movies"  "reading"
# 
# $p3$lang
# $p3$lang$r
# [1] 1
# 
# $p3$lang$cpp
# [1] 4
# 
# $p3$lang$python
# [1] 2
użytkownik2030503
źródło
jak można usunąć elementy python lub lang w tym przykładzie?
Arthur Yip,
9

Nie wiem, czy nadal potrzebujesz odpowiedzi na to pytanie, ale z mojego ograniczonego (3-tygodniowego samouczenia R) doświadczenia z R wynika, że ​​użycie NULLzadania jest w rzeczywistości złe lub nieoptymalne, szczególnie jeśli dynamicznie aktualizujesz lista w czymś w rodzaju pętli for.

Aby być bardziej precyzyjnym, używając

myList[[5]] <- NULL

wyrzuci błąd

myList [[5]] <- NULL: zamiennik ma długość zero

lub

dostarczono więcej elementów niż do wymiany

Stwierdziłem, że działa bardziej konsekwentnie

myList <- myList[[-5]]
użytkownik2035799
źródło
1
Dobra odpowiedź! Myślę jednak, że [[-5]]powinny to być pojedyncze nawiasy kwadratowe, w przeciwnym razie odznaczasz tylko zawartość tego elementu listy, a nie sam element. Cóż, przynajmniej użycie podwójnych nawiasów kwadratowych daje mi ten błąd: „spróbuj wybrać więcej niż jeden element”. Co działa na mnie wtedy: myList <- myList[-5].
n1k31t4
4

Chciałem tylko szybko dodać (bo nie widziałem tego w żadnej z odpowiedzi), że dla nazwanej listy możesz to zrobić l["name"] <- NULL. Na przykład:

l <- list(a = 1, b = 2, cc = 3)
l['b'] <- NULL
Alexey Shiklomanov
źródło
4

Użyj -(znak ujemny) wraz z pozycją elementu, na przykład jeśli trzeci element ma zostać usunięty, użyj go jakoyour_list[-3]

Wejście

my_list <- list(a = 3, b = 3, c = 4, d = "Hello", e = NA)
my_list
# $`a`
# [1] 3

# $b
# [1] 3

# $c
# [1] 4

# $d
# [1] "Hello"

# $e
# [1] NA

Usuń pojedynczy element z listy

 my_list[-3]
 # $`a`
 # [1] 3

 # $b
 # [1] 3

 # $d
 # [1] "Hello"

 # $e
 [1] NA

Usuń wiele elementów z listy

 my_list[c(-1,-3,-2)]
 # $`d`
 # [1] "Hello"

 # $e
 # [1] NA

 my_list[c(-3:-5)]
 # $`a`
 # [1] 3

 # $b
 # [1] 3

 my_list[-seq(1:2)]
 # $`c`
 # [1] 4

 # $d
 # [1] "Hello"

 # $e
 # [1] NA
Sowmya S. Manian
źródło
2

W przypadku list nazwanych przydatne są te funkcje pomocnicze

member <- function(list,names){
    ## return the elements of the list with the input names
    member..names <- names(list)
    index <- which(member..names %in% names)
    list[index]    
}


exclude <- function(list,names){
     ## return the elements of the list not belonging to names
     member..names <- names(list)
     index <- which(!(member..names %in% names))
    list[index]    
}  
aa <- structure(list(a = 1:10, b = 4:5, fruits = c("apple", "orange"
)), .Names = c("a", "b", "fruits"))

> aa
## $a
##  [1]  1  2  3  4  5  6  7  8  9 10

## $b
## [1] 4 5

## $fruits
## [1] "apple"  "orange"


> member(aa,"fruits")
## $fruits
## [1] "apple"  "orange"


> exclude(aa,"fruits")
## $a
##  [1]  1  2  3  4  5  6  7  8  9 10

## $b
## [1] 4 5
DJJ
źródło
0

Używanie lapply i grep:

lst <- list(a = 1:4, b = 4:8, c = 8:10)
# say you want to remove a and c
toremove<-c("a","c")
lstnew<-lst[-unlist(lapply(toremove, function(x) grep(x, names(lst)) ) ) ]
#or
pattern<-"a|c"
lstnew<-lst[-grep(pattern, names(lst))]
Ferroao
źródło
-1

Co powiesz na to? Ponownie, używając indeksów

> m <- c(1:5)
> m
[1] 1 2 3 4 5

> m[1:length(m)-1]
[1] 1 2 3 4

lub

> m[-(length(m))]
[1] 1 2 3 4
RocketRon
źródło
1
m jest wektorem, a nie listą
C8H10N4O2
1
Ta metoda działa dla list, ale OP ma szczęście i prawdopodobnie chce jeszcze kilku nawiasów:m[1:(length(m) - 1)]
Gregor Thomas
-1

jeśli chcesz uniknąć indeksów numerycznych, możesz użyć

a <- setdiff(names(a),c("name1", ..., "namen"))

aby usunąć nazwy namea...namenz. to działa w przypadku list

> l <- list(a=1,b=2)
> l[setdiff(names(l),"a")]
$b
[1] 2

jak również dla wektorów

> v <- c(a=1,b=2)
> v[setdiff(names(v),"a")]
b 
2
Greg Minshall
źródło
-2

Możesz użyć which.

x<-c(1:5)
x
#[1] 1 2 3 4 5
x<-x[-which(x==4)]
x
#[1] 1 2 3 5
Pavidus
źródło
20
To nie jestlist
GSee