Como posso remover um elemento de uma lista?

275

Eu tenho uma lista e quero remover um único elemento dela. Como posso fazer isso?

Tentei procurar o que acho que os nomes óbvios para essa função estariam no manual de referência e não encontrei nada apropriado.

David Locke
fonte
Depende que você deseja removê-lo por valor, por exemplo, "o valor 5" ou por índices / índices "o elemento no índice 5" ou "nos índices c (5: 6,10)? Se você deseja remover por valor e não são duplicatas, você deseja remover apenas as duplicatas, a primeira ou a última ocorrência ou todas? É garantido que a lista contenha seu elemento / índice? Precisamos lidar com o caso em que a lista está vazia? garantir nA é passado (/ excluídos) é a lista garantido para ser plana ou pode ser aninhados Quantas laters profundo??
SMCI
2
setdiff (myList, elementToRemove)
JStrahl

Respostas:

218

Não conheço R, mas um pouco de pesquisa criativa me levou aqui: http://tolstoy.newcastle.edu.au/R/help/05/04/1919.html

A citação principal de lá:

Não encontro documentação explícita para o R sobre como remover elementos das listas, mas tentativa e erro informam

myList [[5]] <- NULL

removerá o quinto elemento e, em seguida, "fechará" o orifício causado pela exclusão desse elemento. Isso é suficiente para os valores do índice, por isso tenho que ter cuidado ao eliminar elementos. Devo trabalhar do final da lista para a frente.

Uma resposta a essa postagem posteriormente no thread declara:

Para excluir um elemento de uma lista, consulte R FAQ 7.1

E a seção relevante do R FAQ diz:

... Não defina x [i] ou x [[i]] como NULL, pois isso removerá o componente correspondente da lista.

O que parece lhe dizer (de uma maneira um pouco atrasada) como remover um elemento.

Espero que ajude, ou pelo menos leve você na direção certa.

Chad Birch
fonte
5
Obrigado, minha lista [i] <- NULL é exatamente a maneira de fazer isso.
1911 David Locke
37
Isto não funcionou para mim. Eu recebo:Error in list[length(list)] <- NULL : replacement has length zero
wfbarksdale
3
Post 's @Aleksandr Levchuck me mostrou que eu estava realmente lidando com um vector e necessário para criar um novo objeto
wfbarksdale
209

Se você não deseja modificar a lista no local (por exemplo, para passar a lista com um elemento removido para uma função), você pode usar a indexação: índices negativos significam "não incluir este elemento".

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

x[-2];       # without 2nd element

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

Além disso, vetores de índice lógicos são úteis:

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

Isso funciona com quadros de dados também:

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
fonte
5
Seu índice lógico funciona apenas se você tiver esse único item "b" em um elemento da lista. Você não pode remover, digamos, x$bdessa maneira, nem o "b" de um elemento da lista x[[2]] = c("b","k") .
Carl Witthoft 31/01
Em relação a itens únicos vs múltiplos: você pode usar %in%para testar vários itens. Não sei ao certo o que você quer dizer com "não é possível remover x $ b" - você quer dizer remover a coluna inteira b?
Florian Jenn
30

Aqui está como remover o último elemento de uma lista em R:

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

Se x pode ser um vetor, você precisará criar um novo objeto:

x <- c("a", "b", "c", "d", "e")
x <- x[-length(x)]
  • Trabalhar para listas e vetores
Aleksandr Levchuk
fonte
@krlmlr: pelo contrário, esta solução é mais geral que a resposta de Florian, pois é polimórfica no tipo de coleção.
precisa saber é o seguinte
@ DanBarowy: Eu estava errado: parece ser uma síntese da resposta de Chad (a aceita) e de Florian ... Um bom resumo, no entanto.
precisa saber é o seguinte
19

Removendo elementos nulos de uma lista em uma única linha:

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

Felicidades

Sukhi
fonte
2
Esse código quebra quando xhá uma lista vazia. Use compactfrom plyrpara esta tarefa.
Richie Cotton
Além disso, se não houver nulos na lista, -(which(sapply(x,is.null),arr.ind=TRUE))retornos named integer(0)que eliminarão completamente essa linha.
user3055034
18

Gostaria de acrescentar que, se for uma lista nomeada, você pode simplesmente usar within.

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

Então você pode substituir a lista original

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

para remover o elemento nomeado ada lista l.

Kim
fonte
1
Para fazer múltiplawithin(l, rm(a, b))
Vlad
16

Se você possui uma lista nomeada e deseja remover um elemento específico, pode tentar:

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

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

Isso fará com que uma lista lstcom os elementos a, b, c. A segunda linha remove o elemento bdepois de verificar se ele existe (para evitar o problema @hjv mencionado).

ou melhor:

lst$b <- NULL

Dessa forma, não é um problema tentar excluir um elemento inexistente (por exemplo lst$g <- NULL)

alko989
fonte
10

Existe o pacote rlist ( http://cran.r-project.org/web/packages/rlist/index.html ) para lidar com vários tipos de operações de lista.

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

Resulta em:

# $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
user2030503
fonte
como remover os itens python ou lang neste exemplo?
Arthur Yip
9

Não sei se você ainda precisa de uma resposta para isso, mas eu descobri pela minha experiência limitada (3 semanas de autodidata R) com R que, usar a NULLtarefa está realmente errado ou não é ideal, especialmente se você estiver atualizando dinamicamente uma lista em algo como um loop for.

Para ser mais preciso, use

myList[[5]] <- NULL

jogará o erro

myList [[5]] <- NULL: a substituição tem comprimento zero

ou

mais elementos fornecidos do que existem para substituir

O que eu achei para trabalhar de forma mais consistente é

myList <- myList[[-5]]
user2035799
fonte
1
Boa resposta! No entanto, acho que [[-5]]devem ser colchetes simples, caso contrário, você está desmarcando apenas o conteúdo desse elemento da lista, não o próprio elemento. Bem, pelo menos o uso de colchetes duplos me dá o seguinte erro: "tentativa de selecionar mais de um elemento". O que funciona para mim foi, em seguida: myList <- myList[-5].
N1k31t4
4

Só queria adicionar rapidamente (porque não o vi em nenhuma das respostas) que, para uma lista nomeada, você também pode fazer l["name"] <- NULL. Por exemplo:

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

Use -(sinal negativo) junto com a posição do elemento, por exemplo, se o terceiro elemento for removido, use-o comoyour_list[-3]

Entrada

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

Remover elemento único da lista

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

 # $b
 # [1] 3

 # $d
 # [1] "Hello"

 # $e
 [1] NA

Remover vários elementos da lista

 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
fonte
2

No caso de listas nomeadas, acho úteis essas funções auxiliares

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
fonte
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))]
Ferroao
fonte
-1

Que tal agora? Novamente, usando índices

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

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

ou

> m[-(length(m))]
[1] 1 2 3 4
RocketRon
fonte
1
m é um vector, não uma lista
C8H10N4O2
1
O método não funciona para listas, mas OP é sorte e provavelmente quer um pouco mais de parênteses:m[1:(length(m) - 1)]
Gregor Thomas
-1

se você quiser evitar índices numéricos, pode usar

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

excluir nomes namea...namende a. isso funciona para listas

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

bem como para vetores

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

Você pode usar which.

x<-c(1:5)
x
#[1] 1 2 3 4 5
x<-x[-which(x==4)]
x
#[1] 1 2 3 5
Pavidus
fonte
20
Isso não é umlist
GSee