Як я можу видалити елемент зі списку?


275

У мене є список, і я хочу видалити з нього один елемент. Як я можу це зробити?

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


Залежить, чи ви хочете вилучити його за значенням, наприклад, "значення 5", або за індексом / індексами "елемент в індексі 5" або "за індексами c (5: 6,10)? Якщо ви хочете видалити за значенням і там є дублікатами, тоді ви хочете видалити лише дублікати, перше чи останнє виникнення чи всі? Чи гарантовано, що список містить ваш елемент / індекс? Чи потрібно нам обробляти випадок, коли список порожній? Чи потрібно нам переконайтеся, що NA передано (/ виключено)? Чи гарантовано список буде рівним чи його можна вкладати? Скільки пізніх глибин?
smci

2
setdiff (myList, elementToRemove)
JStrahl

Відповіді:


218

Я взагалі не знаю R, але сюди привели мене трохи творчого гуглінгу: http://tolstoy.newcastle.edu.au/R/help/05/04/1919.html

Ключова цитата звідти:

Я не знаходжу явної документації для R про те, як видалити елементи зі списків, але пробні та помилкові мені підказують

myList [[5]] <- NULL

видалить 5-й елемент, а потім "закриє" дірку, викликану видаленням цього елемента. Це завищує значення індексу, тож я маю бути обережним при скиданні елементів. Я повинен працювати від задньої частини списку до переду.

Відповідь на цей пост пізніше в потоці станів:

Щоб видалити елемент списку, див. R FAQ 7.1

І у відповідному розділі поширених запитань R йдеться:

... Не встановлюйте x [i] або x [[i]] на NULL, оскільки це видалить відповідний компонент зі списку.

Який, здається, говорить вам (дещо назад), як видалити елемент.

Сподіваюся, що це допоможе або принаймні приведе вас у правильному напрямку.


5
Дякую, мій список [i] <- NULL - саме такий спосіб зробити.
Девід Локк

37
Це для мене не вийшло. Я отримую:Error in list[length(list)] <- NULL : replacement has length zero
wfbarksdale

3
Повідомлення @Laksandr Levchuck показало мені, що я насправді маю справу з вектором і мені потрібно створити новий об'єкт
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

4
Ваш логічний індекс працює лише в тому випадку, якщо у вас є елемент "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: навпаки, це рішення є більш загальним, ніж відповідь Флоріана, оскільки є поліморфним у типі колекції.
Dan Barowy

@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)яких повністю випаде з цього рядка.
користувач3055034

17

Я хотів би додати, що якщо це названий список, ви можете просто використовувати 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)


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

як можна видалити елементи python або lang у цьому прикладі?
Артур Іп

9

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

Якщо бути більш точним, використовуючи

myList[[5]] <- NULL

викине помилку

myList [[5]] <- NULL: заміна має нульову довжину

або

більше елементів, ніж є для заміни

Що я виявив працювати більш послідовно

myList <- myList[[-5]]

1
Хороша відповідь! Однак я думаю, що [[-5]]мають бути одиничні квадратні дужки, інакше ви скасуєте вибір лише вмісту цього елемента списку, а не самого елемента. Ну, принаймні використання подвійних квадратних дужок дає мені таку помилку: "спроба вибрати більше одного елемента". Що працює для мене тоді: myList <- myList[-5].
n1k31t4

4

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

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

4

Використовуйте -(Негативний знак) разом з положенням елемента, наприклад, якщо потрібно видалити 3-й елемент, використовуйте його як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

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))]

-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

1
m - вектор, а не список
C8H10N4O2

1
Метод працює для списків, але ОП пощастило і, ймовірно, хоче ще кілька дужок: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

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.