R logo

NAの取り扱いシリーズ第4段。

 

今回は、他のデータから推測して補完する方法。

よく使われるのは、中央値(median)で補完する方法。

場合によっては平均値(mean)でもいいけど、中央値の方が外れ値の影響を受けにくいので、中央値の方が好まれる。

群の中央値で補完 (1)

例えば下記のようなデータフレーム(Dat)で、"Employees" のデータが2箇所欠損している。

# NA値のある行を全部抽出
            Dat[!complete.cases(Dat), ]
r-na-3-1

補完したい欠損値のみ抽出

!complete.cases()

だと、NA値のあるデータを全て抽出してくるけど、今回は "Employees" にNA値がある行だけを選択したいので、

# "Employees" がNAのデータを抽出
Dat[!complete.cases(Dat$Employees), ]
r-na-3-2

Industry の列を見ると、

  • Retail
  • Financial Services

と異なっている。

両方とも全体の中央値で補完してもいいけど、異なる Industry では Employees の数も違うはず。

なので、

  • "Retail" の "Employees" は "Retail" の中央値で
  • "Financial Services" の "Employees" は "Financial Services" の中央値で

補完した方が、より現実的な値に近くなるはず。

とゆーことで、まずは "Industry" が "Retail" & "Industry" が "NA" の行を抽出。

# "Employees" がNAで、"Industry" が "Retail" のデータを抽出
    Dat[is.na(Dat$Employees) & Dat$Industry=="Retail", ]
r-na-3-3

群の中央値の求め方

"Employees"の列全体の中央値なら、

# 全体の中央値
median(Dat$Employees, na.rm = T)

で求められる。

median(Dat$Employees)

だけにすると、"NA" 値を含む列は全て "NA" で返ってしまうので、

na.rm = T

で「NA値を除く」と指定しておく。

 

"Industry" が "Retail" の行の "Employees" の中央値は、下記で出す。

# "Industry" が "Retail" のデータの "Employees" の中央値
median(Dat[Dat$Industry=="Retail", "Employees"], na.rm = T)

 

全体の "Employees" の中央値は "56"、"Industry" が "Retail" の行の "Employees" の中央値は、"28" という結果が返ってきた。

r-na-3-4

欠損値の部分に群の中央値を入れる

最初に出していた "Industry" が "Retail" & "Industry" が "NA" の行を抽出。

# "Employees" がNAで、"Industry" が "Retail" のデータを抽出
Dat[is.na(Dat$Employees) & Dat$Industry=="Retail", ]
r-na-3-3

で、ここの "Employees" の列に "群の中央値: 28" を入れたいので、

# "Retail" の中央値で補完する
Dat[!complete.cases(Dat$Employees & Dat$Industry=="Retail"), "Employees"] <- median(Dat[Dat$Industry=="Retail", "Employees"], na.rm=T)

 

これで補完されたのだけれど、画面上に現れるわけではないので、該当する行(今回は3行目)を抽出して確認してみると、

# 確認
Dat[3,]

ちゃんと挿入されてた。

群の中央値で補完する(2)

中央値で補完はいいけど、より正確に計算できる場合はその計算によって求めた方がいい。

例えば、下記データフレーム(Dat)で、"Expenses" の値を補完しようと思うけど、

r-na-median-2-1

17行目の "Expenses" は、両隣の "Revenue" と "Profit" から計算できる。

なので、

  1. 他の方法で計算可能な行を除いて NA行を抽出
  2. 群ごとに中央値を求めて代入

という方法をとる。

# Dat$Profit が NA じゃないデータを抽出
Dat[!complete.cases(Dat) & is.na(Dat$Profit),]

すると、8行目と44行目のデータのみが抽出された。

r-na-median-2-2

 

で、ver.1のときと同じように、それぞれのIndustryの中央値で補完する。

今回は両方とも "Construction" なので、

# "Industry" が "Construction" の列の中央値を算出
median(Dat[Dat$Industry=="Construction","Expenses"], na.rm = TRUE)

"4506976" という値が返ってきた。

r-na-median-2-3

これを、最初に抽出した NA のデータに代入する。

# "Industry" が "Construction" の列の中央値を算出
Construction_Expenses_median <- median(Dat[Dat$Industry=="Construction" ,"Expenses"], na.rm=TRUE)
Dat[!complete.cases(Dat) & is.na(Dat$Profit),"Expenses"] <- Construction_Expenses_median

ちゃんと代入されたか、8行目と44行目のデータを確認してみる。

# 確認
Dat[c(8,44),]
r-na-median-2-4

ちゃんと代入されていた。

r-na-3-5

 

 

References

Udemy-English