如何从列表中删除元素?

我有一个列表,我想从其中删除一个元素。我该怎么做呢?

我试着在参考手册中查找我认为这个函数的明显名称,但我没有找到任何合适的名称。

614795 次浏览

我根本不知道R,但一点创造性的谷歌搜索把我带到了这里:http://tolstoy.newcastle.edu.au/R/help/05/04/1919.html

关键的一句话是:

我没有找到关于R如何从列表中删除元素的明确文档,但试验和错误告诉我

myList[[5]] <- NULL

将删除第5个元素,然后“关闭”由删除该元素引起的孔。这影响了下标值,所以我在删除元素时要小心。我必须从名单的后面到前面。

A 稍后回复那篇文章声明:

删除列表中的元素,请参见R FAQ 7.1

R FAQ的相关部分表示:

... 不要将x[i]或x[[i]]设置为NULL,因为这将从列表中删除相应的组件。

它似乎(以一种有点倒退的方式)告诉您如何删除一个元素。

希望这对你有所帮助,或者至少指引你走向正确的方向。

如果你不想就地修改列表(例如,将一个元素传递给一个函数),你可以使用索引:负索引表示“不包括这个元素”。

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


x[-2];       # without 2nd element


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

同样,逻辑索引向量也很有用:

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

这也适用于数据框架:

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

单行从列表中删除Null元素:

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

干杯

下面是如何删除R中列表的最后一个元素:

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

如果x可能是一个向量,那么你需要创建一个新对象:

x <- c("a", "b", "c", "d", "e")
x <- x[-length(x)]
  • 适用于列表向量

你可以使用which

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

如果你有一个命名列表,想要删除一个特定的元素,你可以尝试:

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


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

这将创建一个包含元素abc的列表lst。第二行在检查元素b是否存在后删除它(以避免@hjv提到的问题)。

或更好:

lst$b <- NULL

这样,尝试删除一个不存在的元素(例如lst$g <- NULL)就不是问题了。

rlist包(http://cran.r-project.org/web/packages/rlist/index.html)用于处理各种列表操作。

例子(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"))

结果:

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

不知道你是否还需要回答这个问题,但我从我有限的(3周的自学R) R经验中发现,使用NULL赋值实际上是错误的或次优的,特别是当你在动态更新一个列表时,比如for循环。

更准确地说,使用

myList[[5]] <- NULL

会抛出错误

myList[[5]] <- NULL:替换长度为0

供应的元素多于可供替换的元素

我发现更有效的方法是

myList <- myList[[-5]]

这个怎么样?还是使用下标

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


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

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

只是想快速添加(因为我在任何答案中都没有看到它),对于命名列表,你也可以执行l["name"] <- NULL。例如:

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

使用lapply和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))]

在命名列表的情况下,我发现这些helper函数很有用

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

我想补充的是,如果它是命名的列表,你可以简单地使用within

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

所以你可以覆盖原来的列表

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

从列表l中删除名为a的元素。

使用-(负号)和元素的位置,例如,如果要删除第三个元素,则使用your_list[-3]

输入

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

从列表中删除单个元素

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


# $b
# [1] 3


# $d
# [1] "Hello"


# $e
[1] NA

从列表中删除多个元素

 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

如果您想避免数字索引,可以使用

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

从a中删除名字namea...namen,这适用于列表

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

对于向量也是一样

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

你也可以使用magrittr包的extract函数从列表中删除一个列表项。

a <- seq(1,5)
b <- seq(2,6)
c <- seq(3,7)
l <- list(a,b,c)


library(magrittr)


extract(l,-1) #simple one-function method
[[1]]
[1] 2 3 4 5 6


[[2]]
[1] 3 4 5 6 7

这里有一个简单的解决方案,可以使用底数r。它从原始的数字列表中删除数字5。您可以使用相同的方法从列表中删除任何您想要的元素。

#the original list
original_list = c(1:10)


#the list element to remove
remove = 5


#the new list (which will not contain whatever the `remove` variable equals)
new_list = c()


#go through all the elements in the list and add them to the new list if they don't equal the `remove` variable
counter = 1
for (n in original_list){
if (n != ){
new_list[[counter]] = n
counter = counter + 1
}
}

new_list变量不再包含5。

new_list
# [1]  1  2  3  4  6  7  8  9 10

purrr包中有几个未被提及的选项:

pluckassign_in可以很好地处理嵌套值,你可以使用名称和/或索引的组合来访问它:

library(purrr)


l <- list("a" = 1:2, "b" = 3:4, "d" = list("e" = 5:6, "f" = 7:8))


# select values (by name and/or index)
all.equal(pluck(l, "d", "e"), pluck(l, 3, "e"), pluck(l, 3, 1))
[1] TRUE


# or if element location stored in a vector use !!!
pluck(l, !!! as.list(c("d", "e")))
[1] 5 6


# remove values (modifies in place)
pluck(l, "d", "e") <- NULL


# assign_in to remove values with name and/or index (does not modify in place)
assign_in(l, list("d", 1), NULL)
$a
[1] 1 2


$b
[1] 3 4


$d
$d$f
[1] 7 8


或者你可以通过赋值zap()NULL来使用modify_list删除值:

all.equal(list_modify(l, a = zap()), list_modify(l, a = NULL))
[1] TRUE

可以使用带有discardkeep的谓词函数删除或保留元素:

# remove numeric elements
discard(l, is.numeric)
$d
$d$e
[1] 5 6


$d$f
[1] 7 8


# keep numeric elements
keep(l, is.numeric)
$a
[1] 1 2


$b
[1] 3 4