リストから要素を削除するにはどうすればよいですか?


275

リストがあり、そこから1つの要素を削除したい。これどうやってするの?

この関数のわかりやすい名前がリファレンスマニュアルにあると思いますが、適切なものは何も見つかりませんでした。


「値5」などの値で削除しますか、またはインデックス/インデックス「インデックス5の要素」または「インデックスc(5:6,10)でインデックスを削除しますか?値で削除したい場合は、重複している場合、重複のみ、最初または最後の出現、またはすべてを削除しますか?リストに要素/インデックスが含まれていることが保証されますか?リストが空の場合を処理する必要がありますか? NAが確実に渡される(/除外される)か?リストはフラットであることが保証されているか、それとも入れ子にできるか?深さはいくつですか?
smci

2
setdiff(myList、elementToRemove)
JStrahl '19

回答:


218

私は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はまさにその方法です。
David Locke

37
これは私にはうまくいきませんでした。私が得る:Error in list[length(list)] <- NULL : replacement has length zero
wfbarksdale

3
@Aleksandr 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

5
論理インデックスは、リスト要素にその単一のアイテム「b」がある場合にのみ機能します。たとえば、 x$bそのように削除したり、リスト要素から「b」を削除したりすることはできませんx[[2]] = c("b","k")
Carl Witthoft、2018年

単一のアイテムと複数のアイテムについて:複数のアイテム%in%に対するテストに使用できます。「x $ bを削除できません」という意味がわかりませんb。列全体を削除するということですか?
Florian Jenn

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:逆に、このソリューションはコレクションのタイプが多態的であるため、Florianの回答よりも一般的です。
Dan Barowy 14

@DanBarowy:私は間違っていました:これはチャドの答え(承認されたもの)とフロリアンの答えの合成のようです...しかし、良い簡単な要約です。
krlmlr 2014

19

リストからNull要素を1行で削除する:

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

乾杯


2
xが空のリストの場合、このコードは機能しません。代わりに、このタスクにはcompactfrom plyrを使用してください。
リッチーコットン

また、リストにnullがない場合は、その行を完全に削除するを-(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))
Vlad

16

名前付きリストがあり、特定の要素を削除したい場合は、次のことを試してください。

lst <- list(a = 1:4, b = 4:8, c = 8:10)

if("b" %in% names(lst)) lst <- lst[ - which(names(lst) == "b")]

これは、リストを作るlst要素とabc。2行目は、要素の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の項目をどのように削除できますか?
Arthur Yip

9

これに対する答えがまだ必要かどうかはわかりませんが、NULL特に動的に更新している場合は、割り当ての使用が実際には間違っているか、最適ではないという、Rの限られた(3週間の自己学習R)経験からわかったforループのようなリスト。

より正確に言うと、

myList[[5]] <- NULL

エラーをスローします

myList [[5]] <-NULL:置換の長さがゼロです

または

交換するよりも多くの要素が提供されています

私がより一貫して機能することがわかったのは

myList <- myList[[-5]]

1
いい答えだ!ただし、は[[-5]]単一の角括弧である必要があると思います。そうでない場合は、要素自体ではなく、そのリスト要素のコンテンツのみを選択解除します。少なくとも、二重の角括弧を使用すると、「複数の要素を選択しようとしています」というエラーが表示されます。そのとき私にとってうまくいったのは:myList <- myList[-5]でした。
n1k31t4 2016年

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
この方法は、リストのための作業を行いますが、OPはラッキーで、おそらくいくつかのより多くの括弧を望んでいる:m[1:(length(m) - 1)]
グレゴール・トーマス

-1

数値インデックスを避けたい場合は、

a <- setdiff(names(a),c("name1", ..., "namen"))

namea...namenaから名前を削除します。これはリストで機能します

> 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

20
それはaではありませんlist
GSee
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.