У меня есть список, и я хочу удалить из него один элемент. Как я могу это сделать?
Я попытался найти то, что, как мне кажется, явные названия для этой функции будут в справочном руководстве, и я не нашел ничего подходящего.
У меня есть список, и я хочу удалить из него один элемент. Как я могу это сделать?
Я попытался найти то, что, как мне кажется, явные названия для этой функции будут в справочном руководстве, и я не нашел ничего подходящего.
Ответы:
Я вообще не знаю R, но немного творческого поиска привели меня сюда: http://tolstoy.newcastle.edu.au/R/help/05/04/1919.html
Ключевая цитата оттуда:
Я не нахожу явной документации для R о том, как удалять элементы из списков, но метод проб и ошибок подсказывает мне
myList [[5]] <- NULL
удалит 5-й элемент, а затем "закроет" дыру, вызванную удалением этого элемента. Это превышает значения индекса, поэтому я должен быть осторожен при отбрасывании элементов. Я должен работать от конца списка до фронта.
Ответ на этот пост позже в потоке состояний:
Для удаления элемента списка, см. R FAQ 7.1
И соответствующий раздел R FAQ содержит следующее:
... Не устанавливайте x [i] или x [[i]] в NULL, потому что это удалит соответствующий компонент из списка.
Который, кажется, говорит вам (несколько назад), как удалить элемент.
Надеюсь, что это помогает, или, по крайней мере, ведет вас в правильном направлении.
Error in list[length(list)] <- NULL : replacement has length zero
Если вы не хотите изменять список на месте (например, для передачи списка с удаленным элементом в функцию), вы можете использовать индексирование: отрицательные индексы означают «не включать этот элемент».
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
x$b
таким образом, а также не можете удалить «b» из элемента списка x[[2]] = c("b","k")
.
%in%
для тестирования нескольких предметов. Я не уверен, что вы подразумеваете под «не могу удалить x $ b» - вы имеете в виду удаление всего столбца b
?
Вот как удалить последний элемент списка в R:
x <- list("a", "b", "c", "d", "e")
x[length(x)] <- NULL
Если x может быть вектором, вам нужно создать новый объект:
x <- c("a", "b", "c", "d", "e")
x <- x[-length(x)]
Удаление пустых элементов из списка в одну строку:
x=x[-(which(sapply(x,is.null),arr.ind=TRUE))]
ура
x
пустой список. Используйте compact
от plyr
для этой задачи вместо.
-(which(sapply(x,is.null),arr.ind=TRUE))
возвращается named integer(0)
, что полностью отбрасывает эту строку.
Я хотел бы добавить, что если это именованный список, вы можете просто использовать within
.
l <- list(a = 1, b = 2)
> within(l, rm(a))
$b
[1] 2
Таким образом, вы можете переписать оригинальный список
l <- within(l, rm(a))
удалить элемент с именем a
из списка l
.
within(l, rm(a, b))
Если у вас есть именованный список и вы хотите удалить определенный элемент, вы можете попробовать:
lst <- list(a = 1:4, b = 4:8, c = 8:10)
if("b" %in% names(lst)) lst <- lst[ - which(names(lst) == "b")]
Это позволит сделать список lst
с элементами a
, b
, c
. Вторая строка удаляет элемент 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
Не знаю, нужен ли вам ответ на этот вопрос, но из моего ограниченного (самообучающегося R) опыта работы с R я узнал, что использование NULL
назначения на самом деле неправильно или неоптимально, особенно если вы динамически обновляете список в чем-то вроде цикла for.
Чтобы быть более точным, используя
myList[[5]] <- NULL
выдаст ошибку
myList [[5]] <- NULL: длина замены равна нулю
или
поставлено больше элементов, чем нужно заменить
То, что я нашел, чтобы работать более последовательно,
myList <- myList[[-5]]
[[-5]]
должны быть одинарные квадратные скобки, в противном случае вы отменяете выделение только содержимого этого элемента списка, а не самого элемента. Ну, по крайней мере, использование двойных квадратных скобок дает мне такую ошибку: «попытка выбрать более одного элемента». Что работает для меня тогда: myList <- myList[-5]
.
Просто хотел быстро добавить (потому что я не видел этого ни в одном из ответов), что для именованного списка вы также можете сделать l["name"] <- NULL
. Например:
l <- list(a = 1, b = 2, cc = 3)
l['b'] <- NULL
Используйте -
(отрицательный знак) вместе с положением элемента, например, если третий элемент должен быть удален, используйте его как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
В случае именованных списков я нахожу эти вспомогательные функции полезными
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
Как насчет этого? Опять же, используя индексы
> 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
m[1:(length(m) - 1)]
если вы хотите избежать числовых индексов, вы можете использовать
a <- setdiff(names(a),c("name1", ..., "namen"))
удалить имена 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