У мене є список, і я хочу видалити з нього один елемент. Як я можу це зробити?
Я спробував знайти те, що, на мою думку, очевидні назви цієї функції були б у посібнику, і я не знайшов нічого підходящого.
У мене є список, і я хочу видалити з нього один елемент. Як я можу це зробити?
Я спробував знайти те, що, на мою думку, очевидні назви цієї функції були б у посібнику, і я не знайшов нічого підходящого.
Відповіді:
Я взагалі не знаю 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, оскільки це видалить відповідний компонент зі списку.
Який, здається, говорить вам (дещо назад), як видалити елемент.
Сподіваюся, що це допоможе або принаймні приведе вас у правильному напрямку.
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
Не знаю, чи все ще вам потрібна відповідь на це, але я виявив з мого обмеженого (3 тижні вартістю самонавчання R) з R, що використання NULL
завдання фактично неправильне або неоптимальне, особливо якщо ви динамічно оновлюєтесь. список у чимось на зразок for-loop.
Якщо бути більш точним, використовуючи
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
Використовуйте -
(Негативний знак) разом з положенням елемента, наприклад, якщо потрібно видалити 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
У випадку названих списків я вважаю корисними ці допоміжні функції
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