Come posso rimuovere un elemento da un elenco?


275

Ho un elenco e desidero rimuovere un singolo elemento da esso. Come posso fare questo?

Ho provato a cercare ciò che penso che i nomi ovvi per questa funzione sarebbero nel manuale di riferimento e non ho trovato nulla di appropriato.


Dipende vuoi rimuoverlo per valore, ad es. "Il valore 5", o per indice / indici "l'elemento all'indice 5" o "agli indici c (5: 6,10)? Se vuoi rimuovere per valore e lì sono duplicati, quindi vuoi rimuovere solo i duplicati, la prima o l'ultima occorrenza o tutti? È garantito che l'elenco contenga il tuo elemento / indice? Dobbiamo gestire il caso in cui l'elenco è vuoto? Dobbiamo assicurarsi che NA sia passato (/ escluso)? L'elenco è sicuro di essere piatto o può essere nidificato? Quanti laters è profondo?
smci

2
setdiff (myList, elementToRemove)
JStrahl

Risposte:


218

Non conosco affatto R, ma un po 'di googling creativo mi ha portato qui: http://tolstoy.newcastle.edu.au/R/help/05/04/1919.html

La citazione chiave da lì:

Non trovo documentazione esplicita per R su come rimuovere elementi dagli elenchi, ma prova ed errore mi dice

myList [[5]] <- NULL

rimuoverà il 5 ° elemento e quindi "chiuderà" il foro causato dalla cancellazione di tale elemento. Questo soffoca i valori dell'indice, quindi devo stare attento a far cadere gli elementi. Devo lavorare dal retro dell'elenco in primo piano.

Una risposta a quel post in seguito nel thread indica:

Per eliminare un elemento di un elenco, vedere R FAQ 7.1

E la sezione pertinente della R FAQ dice:

... Non impostare x [i] o x [[i]] su NULL, poiché ciò rimuoverà il componente corrispondente dall'elenco.

Il che sembra dirti (in modo un po 'arretrato) come rimuovere un elemento.

Spero che ti aiuti, o almeno ti porti nella giusta direzione.


5
Grazie, mylist [i] <- NULL è esattamente il modo di farlo.
David Locke,

37
Questo non ha funzionato per me. Ottengo:Error in list[length(list)] <- NULL : replacement has length zero
wfbarksdale,

3
Il post di @Aleksandr Levchuck mi ha mostrato che avevo effettivamente a che fare con un vettore e che dovevo creare un nuovo oggetto
wfbarksdale,

209

Se non si desidera modificare l'elenco sul posto (ad esempio per passare l'elenco con un elemento rimosso a una funzione), è possibile utilizzare l'indicizzazione: gli indici negativi significano "non includere questo elemento".

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

x[-2];       # without 2nd element

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

Inoltre, i vettori di indice logici sono utili:

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

Funziona anche con i frame di dati:

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

5
Il tuo indice logico funziona solo se hai quel singolo elemento "b" in un elemento dell'elenco. Non è possibile rimuovere, diciamo, in x$bquesto modo, né è possibile rimuovere "b" da un elemento dell'elenco x[[2]] = c("b","k") .
Carl Witthoft,

Per quanto riguarda gli articoli singoli o multipli: è possibile utilizzare %in%per il test su più articoli. Non sono sicuro di cosa intendi con "impossibile rimuovere x $ b" - intendi rimuovere l'intera colonna b?
Florian Jenn,

30

Ecco come rimuovere l'ultimo elemento di un elenco in R:

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

Se x potrebbe essere un vettore, dovrai creare un nuovo oggetto:

x <- c("a", "b", "c", "d", "e")
x <- x[-length(x)]
  • Lavora per elenchi e vettori

@krlmlr: al contrario, questa soluzione è più generale della risposta di Florian, in quanto polimorfica nel tipo di raccolta.
Dan Barowy,

@DanBarowy: mi sbagliavo: questa sembra essere una sintesi della risposta di Chad (quella accettata) e di Florian ... Un buon breve riassunto, però.
krlmlr,

19

Rimozione di elementi null da un elenco su una sola riga:

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

Saluti


2
Questo codice si interrompe quando xè un elenco vuoto. Utilizzare compactda plyrper questo compito, invece.
Richie Cotton,

Inoltre, se non ci sono valori nulli nell'elenco, -(which(sapply(x,is.null),arr.ind=TRUE))restituisce named integer(0)che eliminerà completamente quella riga.
user3055034

18

Vorrei aggiungere che se è un elenco di nomi che puoi semplicemente usare within.

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

Quindi puoi sovrascrivere l'elenco originale

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

per rimuovere l'elemento denominato adall'elenco l.


1
Per fare piùwithin(l, rm(a, b))
Vlad

16

Se hai un elenco di nomi e desideri rimuovere un elemento specifico, puoi provare:

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

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

Questo farà una lista lstcon gli elementi a, b, c. La seconda riga rimuove l'elemento bdopo aver verificato che esiste (per evitare il problema menzionato da @hjv).

o meglio:

lst$b <- NULL

In questo modo non è un problema provare a eliminare un elemento inesistente (ad es. lst$g <- NULL)


10

C'è il pacchetto rlist ( http://cran.r-project.org/web/packages/rlist/index.html ) per gestire vari tipi di operazioni dell'elenco.

Esempio ( 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"))

Risultati in:

# $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

come si possono rimuovere gli elementi python o lang in questo esempio?
Arthur Yip,

9

Non so se hai ancora bisogno di una risposta a questo, ma ho scoperto dalla mia esperienza limitata (3 settimane di autoapprendimento R) con R che, l'uso del NULLcompito è in realtà sbagliato o subottimale, specialmente se stai aggiornando dinamicamente un elenco in qualcosa come un for-loop.

Per essere più precisi, usando

myList[[5]] <- NULL

genererà l'errore

myList [[5]] <- NULL: la sostituzione ha lunghezza zero

o

più elementi forniti di quanti ce ne siano da sostituire

Ciò che ho scoperto per funzionare in modo più coerente è

myList <- myList[[-5]]

1
Buona risposta! Tuttavia, penso che [[-5]]dovrebbero essere parentesi quadre singole, altrimenti si sta deselezionando solo il contenuto di quell'elemento elenco, non l'elemento stesso. Bene, almeno l'uso di parentesi quadre doppie mi dà questo errore: "tenta di selezionare più di un elemento". Ciò che funziona per me era allora: myList <- myList[-5].
n1k31t4,

4

Volevo solo aggiungere rapidamente (perché non l'ho visto in nessuna delle risposte) che, per un elenco di nomi, puoi anche fare l["name"] <- NULL. Per esempio:

l <- list(a = 1, b = 2, cc = 3)
l['b'] <- NULL

4

Utilizzare -(segno negativo) insieme alla posizione dell'elemento, ad esempio se si desidera rimuovere il 3 ° elemento utilizzarlo comeyour_list[-3]

Ingresso

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

Rimuovi singolo elemento dall'elenco

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

 # $b
 # [1] 3

 # $d
 # [1] "Hello"

 # $e
 [1] NA

Rimuovi più elementi dall'elenco

 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

2

Nel caso delle liste nominate trovo utili quelle funzioni di aiuto

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

0

Usando lapply e 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))]

-1

Cosa ne pensi di questo? Ancora una volta, usando gli indici

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

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

o

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

1
m è un vettore, non un elenco
C8H10N4O2

1
Il metodo funziona per le liste, ma OP è fortunato e probabilmente vuole qualche altra parentesi:m[1:(length(m) - 1)]
Gregor Thomas

-1

se vuoi evitare gli indici numerici, puoi usare

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

per cancellare i nomi namea...namenda a. questo funziona per gli elenchi

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

così come per i vettori

> v <- c(a=1,b=2)
> v[setdiff(names(v),"a")]
b 
2

-2

È possibile utilizzare which.

x<-c(1:5)
x
#[1] 1 2 3 4 5
x<-x[-which(x==4)]
x
#[1] 1 2 3 5

20
Non è unlist
GSee,
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.