Как я могу удалить элемент из списка?

173

У меня есть список, и я хочу удалить из него один элемент. Как я могу это сделать?

Я попытался найти то, что, по моему мнению, очевидные имена этой функции будут в справочном руководстве, и я не нашел ничего подходящего.

  • 0
    Зависит от того, хотите ли вы удалить его по значению, например, «значение 5», или по индексу / индексам «элемент по индексу 5» или «по индексам с (5: 6,10)?» Если вы хотите удалить по значению и там являются дубликатами, то хотите ли вы удалить только дубликаты, первое или последнее вхождение или все? Гарантируется ли, что список содержит ваш элемент / индекс? Нужно ли нам обрабатывать случай, когда список пуст? Нужно ли нам гарантировать, что NA пропущен (/ исключен)? Гарантируется ли, что список будет плоским или может быть вложенным? Сколько будет позже?
  • 1
    setdiff (MyList, elementToRemove)
Теги:
list
indexing

11 ответов

156
Лучший ответ

Я вообще не знаю R, но немного креативный googling привел меня сюда: http://tolstoy.newcastle.edu.au/R/help/05/04/1919.html

Ключевая цитата оттуда:

Я не нашел явной документации для R о том, как удалить элементы из списков, но пробная версия и ошибка говорят мне

myList [[5]] < - NULL

удалит 5-й элемент, а затем "закроет" отверстие, вызванное удалением этого элемента. Это наполняет значения индекса, поэтому я должен быть осторожным при удалении элементов. Я должен работать из задней части списка на передний план.

A ответ на этот пост позже в потоке гласит:

Чтобы удалить элемент списка, см. R FAQ 7.1

И соответствующий раздел часто задаваемых вопросов R говорит:

... Не устанавливайте x [i] или x [[i]] в NULL, потому что это приведет к удалению соответствующего компонента из списка.

Что, кажется, говорит вам (несколько назад), как удалить элемент.

Надеюсь, что это поможет или по крайней мере приведет вас в правильном направлении.

  • 2
    Спасибо, mylist [i] <- NULL - это как раз то, что нужно.
  • 32
    Это не сработало для меня. Я получаю: Error in list[length(list)] <- NULL : replacement has length zero
Показать ещё 1 комментарий
133

Если вы не хотите изменять список на месте (например, для передачи списка с удаленным элементом в функцию), вы можете использовать индексирование: отрицательные индексы означают "не включать этот элемент".

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
  • 1
    Ваш логический индекс работает, только если у вас есть этот единственный элемент «b» в элементе списка. Вы не можете удалить, скажем, x$b таким образом, а также не можете удалить «b» из элемента списка x[[2]] = c("b","k") .
  • 0
    Относительно одного против нескольких элементов: вы можете использовать %in% для тестирования нескольких элементов. Я не уверен, что вы подразумеваете под «не могу удалить x $ b» - вы имеете в виду удаление всего столбца b ?
23

Вот как удалить последний элемент списка в R:

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

Если x может быть вектором, вам нужно будет создать новый объект:

x <- c("a", "b", "c", "d", "e")
x <- x[-length(x)]
  • Работайте для списков и векторов
  • 0
    Особый случай ответа Флориана ...
  • 0
    @krlmlr: напротив, это решение более общее, чем ответ Флориана, поскольку оно полиморфно в типе коллекции.
Показать ещё 1 комментарий
16

Удаление нулевых элементов из списка в одной строке:

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

Приветствия

  • 2
    Этот код прерывается, когда x - пустой список. plyr этого используйте compact от plyr .
  • 0
    Также, если в списке нет нулей, -(which(sapply(x,is.null),arr.ind=TRUE)) возвращает named integer(0) которое полностью -(which(sapply(x,is.null),arr.ind=TRUE)) эту строку.
10

Если у вас есть именованный список и вы хотите удалить определенный элемент, вы можете попробовать:

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)

6

Не знаю, нужен ли вам еще ответ на этот вопрос, но я нашел из своего ограниченного (3-недельного опыта самообучения R) опыта с R, который, используя назначение NULL, на самом деле является неправильным или неоптимальным, особенно если вы динамически обновляете список в чем-то вроде цикла for.

Чтобы быть более точным, используя

myList[[5]] <- NULL

выдает ошибку

myList [[5]] < - NULL: замена имеет нулевую длину

или

добавлено больше элементов, чем есть, чтобы заменить

То, что я нашел для работы более последовательно, -

myList <- myList[[-5]]
  • 1
    Хороший ответ! Однако я думаю, что [[-5]] должны быть в квадратных скобках, в противном случае вы отменяете выбор только содержимого этого элемента списка, а не самого элемента. Ну, по крайней мере, использование двойных квадратных скобок дает мне такую ошибку: «попытка выбрать более одного элемента». Тогда у меня myList <- myList[-5] : myList <- myList[-5] .
6

Там находится пакет 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
  • 0
    Спасибо за ваш комментарий. Я добавил соответствующий код.
  • 0
    Как можно удалить элементы Python или Lang в этом примере?
0

Использование 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))]
0

Просто хотел быстро добавить (потому что я не видел его ни в одном из ответов), что для именованного списка вы также можете сделать l["name"] <- NULL. Например:

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

Как насчет этого? Опять же, используя индексы

> 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
  • 1
    м это вектор, а не список
  • 1
    Метод работает для списков, но OP везет и, вероятно, хочет еще несколько скобок: m[1:(length(m) - 1)]
-1

Вы можете использовать which.

x<-c(1:5)
x
#[1] 1 2 3 4 5
x<-x[-which(x==4)]
x
#[1] 1 2 3 5
  • 18
    Это не list

Ещё вопросы

Сообщество Overcoder
Наверх
Меню