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

275

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

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

Дэвид Лок
источник
Зависит от того, хотите ли вы удалить его по значению, например, «значение 5», или по индексу / индексам «элемент по индексу 5» или «по индексам с (5: 6,10)?» Если вы хотите удалить по значению и там являются дубликатами, то хотите ли вы удалить только дубликаты, первое или последнее вхождение или все? Гарантируется, что список содержит ваш элемент / индекс? Нужно ли нам обрабатывать случай, когда список пуст? Нужно ли нам гарантировать, что NA пропущен (/ исключен)? Гарантируется ли, что список будет плоским или может быть вложенным? Сколько будет позже?
smci
2
setdiff (myList, elementToRemove)
JStrahl

Ответы:

219

Я вообще не знаю 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, потому что это удалит соответствующий компонент из списка.

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

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

Чад берез
источник
5
Спасибо, mylist [i] <- NULL - это как раз то, что нужно.
Дэвид Лок
37
Это не сработало для меня. Я получаю:Error in list[length(list)] <- NULL : replacement has length zero
wfbarksdale
3
Сообщение Александра Левчука показало мне, что я на самом деле имел дело с вектором и мне нужно было создать новый объект
wfbarksdale
209

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

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

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

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

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

x <- c("a", "b", "c", "d", "e")
x <- x[-length(x)]
  • Работа для списков и векторов
Александр Левчук
источник
@krlmlr: напротив, это решение более общее, чем ответ Флориана, так как оно полиморфно в типе коллекции.
Дэн Барови
@DanBarowy: я был не прав: похоже, это синтез ответа Чада (принятого) и ответа Флориана ... Хорошее краткое резюме.
krlmlr
19

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

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

ура

Сухи
источник
2
Этот код ломается, когда xпустой список. Используйте compactот plyrдля этой задачи вместо.
Ричи Коттон
Также, если в списке нет нулей, -(which(sapply(x,is.null),arr.ind=TRUE))возвращается named integer(0), что полностью отбрасывает эту строку.
user3055034
18

Я хотел бы добавить, что если это именованный список, вы можете просто использовать within.

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

Таким образом, вы можете переписать оригинальный список

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

удалить элемент с именем aиз списка l.

Ким
источник
1
Чтобы сделать несколькоwithin(l, rm(a, b))
Влад
16

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

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)

alko989
источник
10

Существует пакет 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
user2030503
источник
Как можно удалить элементы Python или Lang в этом примере?
Артур Йип
9

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

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

myList[[5]] <- NULL

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

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

или

поставлено больше элементов, чем нужно заменить

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

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

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

l <- list(a = 1, b = 2, cc = 3)
l['b'] <- NULL
Алексей Шикломанов
источник
4

Используйте -(отрицательный знак) вместе с положением элемента, например, если третий элемент должен быть удален, используйте его как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
Соумя С. Маниан
источник
2

В случае именованных списков я нахожу эти вспомогательные функции полезными

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
источник
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))]
Ferroao
источник
-1

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

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

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

x<-c(1:5)
x
#[1] 1 2 3 4 5
x<-x[-which(x==4)]
x
#[1] 1 2 3 5
Pavidus
источник
20
Это неlist
GSee