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.
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.
Respostas:
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.
Error in list[length(list)] <- NULL : replacement has length zero
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
x$b
dessa maneira, nem o "b" de um elemento da lista x[[2]] = c("b","k")
.
%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
?
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)]
Removendo elementos nulos de uma lista em uma única linha:
x=x[-(which(sapply(x,is.null),arr.ind=TRUE))]
Felicidades
x
há uma lista vazia. Use compact
from plyr
para esta tarefa.
-(which(sapply(x,is.null),arr.ind=TRUE))
retornos named integer(0)
que eliminarão completamente essa linha.
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 a
da lista l
.
within(l, rm(a, b))
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 lst
com os elementos a
, b
, c
. A segunda linha remove o elemento b
depois 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
)
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
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 NULL
tarefa 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]]
[[-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]
.
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
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
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
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
m[1:(length(m) - 1)]
se você quiser evitar índices numéricos, pode usar
a <- setdiff(names(a),c("name1", ..., "namen"))
excluir nomes namea...namen
de 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