Mam listę i chcę usunąć z niej pojedynczy element. W jaki sposób mogę to zrobić?
Próbowałem sprawdzić, jakie są oczywiste nazwy tej funkcji w podręczniku użytkownika i nie znalazłem nic odpowiedniego.
Mam listę i chcę usunąć z niej pojedynczy element. W jaki sposób mogę to zrobić?
Próbowałem sprawdzić, jakie są oczywiste nazwy tej funkcji w podręczniku użytkownika i nie znalazłem nic odpowiedniego.
Odpowiedzi:
W ogóle nie znam R., ale doprowadziło mnie tutaj trochę kreatywnego googlingu: http://tolstoy.newcastle.edu.au/R/help/05/04/1919.html
Kluczowy cytat stamtąd:
Nie znajduję wyraźnej dokumentacji dla R na temat usuwania elementów z list, ale próby i błędy mówią mi
myList [[5]] <- NULL
usunie piąty element, a następnie „zamknie” dziurę spowodowaną usunięciem tego elementu. To wyczerpuje wartości indeksu, więc muszę uważać na upuszczanie elementów. Muszę pracować od końca listy do przodu.
Odpowiedź na to stanowisko później w wątku stanów:
Aby usunąć element listy, zobacz R FAQ 7.1
Oraz odpowiedniej części R FAQ mówi:
... Nie ustawiaj x [i] lub x [[i]] na NULL, ponieważ spowoduje to usunięcie odpowiedniego komponentu z listy.
Co wydaje się mówić (nieco do tyłu), jak usunąć element.
Nadzieja, która pomaga, a przynajmniej prowadzi cię we właściwym kierunku.
Error in list[length(list)] <- NULL : replacement has length zero
Jeśli nie chcesz modyfikować listy w miejscu (np. Do przekazywania listy z elementem usuniętym do funkcji), możesz użyć indeksowania: ujemne wskaźniki oznaczają „nie uwzględniaj tego elementu”.
x <- list("a", "b", "c", "d", "e"); # example list
x[-2]; # without 2nd element
x[-c(2, 3)]; # without 2nd and 3rd
Przydatne są również wektory indeksu logicznego:
x[x != "b"]; # without elements that are "b"
Działa to również z ramkami danych:
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
ten sposób, ani nie można usunąć „b” z elementu listy x[[2]] = c("b","k")
.
%in%
do testowania wielu przedmiotów. Nie jestem pewien, co rozumiesz przez „nie można usunąć x $ b” - czy masz na myśli usunięcie całej kolumny b
?
Oto jak usunąć ostatni element listy w R:
x <- list("a", "b", "c", "d", "e")
x[length(x)] <- NULL
Jeśli x może być wektorem, musisz utworzyć nowy obiekt:
x <- c("a", "b", "c", "d", "e")
x <- x[-length(x)]
Usuwanie elementów Null z listy w jednym wierszu:
x=x[-(which(sapply(x,is.null),arr.ind=TRUE))]
Twoje zdrowie
x
jest pusta lista. Zamiast tego użyj compact
z plyr
tego zadania.
-(which(sapply(x,is.null),arr.ind=TRUE))
zwraca, named integer(0)
co spowoduje całkowite usunięcie tego wiersza.
Jeśli masz nazwaną listę i chcesz usunąć określony element, możesz spróbować:
lst <- list(a = 1:4, b = 4:8, c = 8:10)
if("b" %in% names(lst)) lst <- lst[ - which(names(lst) == "b")]
Będzie to zrobić listę lst
z elementami a
, b
, c
. Druga linia usuwa elementb
po sprawdzeniu, że istnieje (aby uniknąć wspomnianego problemu @hjv).
albo lepiej:
lst$b <- NULL
W ten sposób nie jest problemem próba usunięcia nieistniejącego elementu (np. lst$g <- NULL
)
Istnieje pakiet rlist ( http://cran.r-project.org/web/packages/rlist/index.html ) do obsługi różnego rodzaju operacji na listach.
Przykład ( 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"))
Prowadzi do:
# $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
Nie wiem, czy nadal potrzebujesz odpowiedzi na to pytanie, ale z mojego ograniczonego (3-tygodniowego samouczenia R) doświadczenia z R wynika, że użycie NULL
zadania jest w rzeczywistości złe lub nieoptymalne, szczególnie jeśli dynamicznie aktualizujesz lista w czymś w rodzaju pętli for.
Aby być bardziej precyzyjnym, używając
myList[[5]] <- NULL
wyrzuci błąd
myList [[5]] <- NULL: zamiennik ma długość zero
lub
dostarczono więcej elementów niż do wymiany
Stwierdziłem, że działa bardziej konsekwentnie
myList <- myList[[-5]]
[[-5]]
powinny to być pojedyncze nawiasy kwadratowe, w przeciwnym razie odznaczasz tylko zawartość tego elementu listy, a nie sam element. Cóż, przynajmniej użycie podwójnych nawiasów kwadratowych daje mi ten błąd: „spróbuj wybrać więcej niż jeden element”. Co działa na mnie wtedy: myList <- myList[-5]
.
Chciałem tylko szybko dodać (bo nie widziałem tego w żadnej z odpowiedzi), że dla nazwanej listy możesz to zrobić l["name"] <- NULL
. Na przykład:
l <- list(a = 1, b = 2, cc = 3)
l['b'] <- NULL
Użyj -
(znak ujemny) wraz z pozycją elementu, na przykład jeśli trzeci element ma zostać usunięty, użyj go jakoyour_list[-3]
Wejście
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
Usuń pojedynczy element z listy
my_list[-3]
# $`a`
# [1] 3
# $b
# [1] 3
# $d
# [1] "Hello"
# $e
[1] NA
Usuń wiele elementów z listy
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
W przypadku list nazwanych przydatne są te funkcje pomocnicze
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
Co powiesz na to? Ponownie, używając indeksów
> m <- c(1:5)
> m
[1] 1 2 3 4 5
> m[1:length(m)-1]
[1] 1 2 3 4
lub
> m[-(length(m))]
[1] 1 2 3 4
m[1:(length(m) - 1)]
jeśli chcesz uniknąć indeksów numerycznych, możesz użyć
a <- setdiff(names(a),c("name1", ..., "namen"))
aby usunąć nazwy namea...namen
z. to działa w przypadku list
> l <- list(a=1,b=2)
> l[setdiff(names(l),"a")]
$b
[1] 2
jak również dla wektorów
> v <- c(a=1,b=2)
> v[setdiff(names(v),"a")]
b
2