超初心者向けのRガイド

f. 分割表(クロス表)の作成

Author

SUGINO Isamu, Build with R4.4.2

Published

December 11, 2024

1 全体の構成

2 二変数の分割表(クロス表)

 調査結果の報告においては,高度な分析をするだけではなく,収集した情報の基本的な集計結果を示す事も重要になる。
しかしRでは,度数分布表(単純集計表などとも呼ばれる)や分割表(連関表,クロス表)について,必要最低限の結果を出力するコマンドが基本で,色々な情報を纏めて提示する為には自分で少し手を加える必要がある。
 以下では,学部学生の演習レヴェルであると便利だと思われる作表方法を紹介する。

2.1 模擬データの作成

 以下は,二つの変数(5件法と4件法,NA混じり)を発生させて,それを(敢えて)データ・フレイムにしている。
何故敢えてデータ・フレイムにしているかと言えば,通常社会調査データはcsvファイルでデータを作成し,それをデータ・フレイムとしてRに読み込んで分析するので,それに近い状態を再現する為である。

n <- 125

q01 <- sample(c(1, 2, 3, 4, 5, NA), 
              size    = n,
              replace = T, 
              prob    = c(.25, .15, .20, .20, .15, .05))
q02 <- sample(c(1, 2, 3, 4, NA),
              size    = n,
              replace = T, 
              prob    = c(.30, .20, .20, .25, .05))

d01 <- data.frame(q01, q02)

 それぞれの変数にアクセスする為には,d01$q01,d01$q02とする事になる。

2.2 二つのカテゴリカル変数の分割表(クロス表)

 社会学でクロス表と呼ぶ事の多い分割表(連関表; contingency table)は二つのカテゴリカル変数の集計表であるが,これを作成するRの関数も table( ) である。table( ) に二つの変数を引数で与えると二変数の分割表になる。一変数の場合と同じく,useNA = オプションを指定しないとNAを除外した表になる。
 table( ) で作成した分割表は表の本体だけであるが,集計結果を示す場合には,行周辺度数・列周辺度数なども併せて表示する。
table( ) で作成された分割表が t01 と云うオブジェクトに格納されているとすると,周辺度数を付加した表は,addmargins( ) 関数にt01を引数として与えれば作成される。
 大抵の場合,度数ではなく相対度数の表も作成したい。関数は一変数の時と同じ prop.table( ) 関数である。
何もオプションを付けないと全体%の表になり,margin = 1 と云うオプションを付けると行%の表になる。margin = 2 では列%の表である。

オブジェクト名は適当であり,自由に付けてよい。

table(d01$q01, d01$q02)
   
     1  2  3  4
  1  9  8  9 11
  2  3  4  2  3
  3  7  3  3  5
  4  7  5  6 12
  5  6  3  1  5
table(d01$q01, d01$q02, useNA = "ifany")
      
        1  2  3  4 <NA>
  1     9  8  9 11    1
  2     3  4  2  3    1
  3     7  3  3  5    0
  4     7  5  6 12    2
  5     6  3  1  5    1
  <NA>  2  0  2  3    1
prop.table(table(d01$q01, d01$q02), margin = 1)
   
             1          2          3          4
  1 0.24324324 0.21621622 0.24324324 0.29729730
  2 0.25000000 0.33333333 0.16666667 0.25000000
  3 0.38888889 0.16666667 0.16666667 0.27777778
  4 0.23333333 0.16666667 0.20000000 0.40000000
  5 0.40000000 0.20000000 0.06666667 0.33333333
prop.table(table(d01$q01, d01$q02), margin = 2)
   
             1          2          3          4
  1 0.28125000 0.34782609 0.42857143 0.30555556
  2 0.09375000 0.17391304 0.09523810 0.08333333
  3 0.21875000 0.13043478 0.14285714 0.13888889
  4 0.21875000 0.21739130 0.28571429 0.33333333
  5 0.18750000 0.13043478 0.04761905 0.13888889
prop.table(table(d01$q01, d01$q02))
   
              1           2           3           4
  1 0.080357143 0.071428571 0.080357143 0.098214286
  2 0.026785714 0.035714286 0.017857143 0.026785714
  3 0.062500000 0.026785714 0.026785714 0.044642857
  4 0.062500000 0.044642857 0.053571429 0.107142857
  5 0.053571429 0.026785714 0.008928571 0.044642857
addmargins(table(d01$q01, d01$q02))
     
        1   2   3   4 Sum
  1     9   8   9  11  37
  2     3   4   2   3  12
  3     7   3   3   5  18
  4     7   5   6  12  30
  5     6   3   1   5  15
  Sum  32  23  21  36 112

with( )round( ) を使い,適宜オブジェクト保存してもう少し見易くしよう。

t01  <- with(d01, table(q01, q02)); t01
   q02
q01  1  2  3  4
  1  9  8  9 11
  2  3  4  2  3
  3  7  3  3  5
  4  7  5  6 12
  5  6  3  1  5
t01T <- with(d01, table(q01, q02, useNA = "ifany")); t01T
      q02
q01     1  2  3  4 <NA>
  1     9  8  9 11    1
  2     3  4  2  3    1
  3     7  3  3  5    0
  4     7  5  6 12    2
  5     6  3  1  5    1
  <NA>  2  0  2  3    1
pr01 <- prop.table(t01, margin = 1) # 行比率
round(pr01*100, 1)
   q02
q01    1    2    3    4
  1 24.3 21.6 24.3 29.7
  2 25.0 33.3 16.7 25.0
  3 38.9 16.7 16.7 27.8
  4 23.3 16.7 20.0 40.0
  5 40.0 20.0  6.7 33.3
pc01 <- prop.table(t01, margin = 2) # 列比率
round(pc01*100, 1)
   q02
q01    1    2    3    4
  1 28.1 34.8 42.9 30.6
  2  9.4 17.4  9.5  8.3
  3 21.9 13.0 14.3 13.9
  4 21.9 21.7 28.6 33.3
  5 18.8 13.0  4.8 13.9
ps01 <- prop.table(t01) # 全体比率
round(ps01*100, 1)
   q02
q01    1    2    3    4
  1  8.0  7.1  8.0  9.8
  2  2.7  3.6  1.8  2.7
  3  6.2  2.7  2.7  4.5
  4  6.2  4.5  5.4 10.7
  5  5.4  2.7  0.9  4.5
addmargins( )による周辺度数
t01m <- addmargins(t01)
t01m
     q02
q01     1   2   3   4 Sum
  1     9   8   9  11  37
  2     3   4   2   3  12
  3     7   3   3   5  18
  4     7   5   6  12  30
  5     6   3   1   5  15
  Sum  32  23  21  36 112

2.3 集計表の項目ラベル

 SPSSでは変数に変数ラベル,値に値ラベルを定義する事が出来,出力を見易くする事が出来るが,Rはこの点については不親切である。
もっとも特に値ラベルについては,実体としての数値が何であるかが分からなくなって却って初心者が間違える場合もあるので,うまく付けないと(或いは適切に表示オプションを設定しないと)便利ではなくなる。
 いずれにせよRでも出力を分かり易くする方法は知っておいた方が良いので,ここでは分割表にラベルを付ける単純な方法を紹介する。
 まずは模擬データを生成する。

性別と教育年数の架空データでデータ・フレイムを作成

n <- 800
sex    <- sample(c(1, 2, 3), 
                 size    = n, 
                 replace = T, 
                 prob    = c(.45, .50, .05))
school <- sample(c(9, 12, 14, 16, 18), 
                 size    = n, 
                 replace = T, 
                 prob    = c(.10, .30, .15, .40, .05))

演習上,男女で分布が異なる方が例示の役に立つので,男性データに改変を加える。

.N <- length(school[sex == 1 & school == 14])
school[sex == 1 & school == 14] <-  school[sex == 1 & school == 14] + 
  sample(c(-2, 0, +2), size = .N, replace = T, prob = c(.5, .1, .4))
d02 <- data.frame(ID = 1:n, sex, school)
head(d02)

このデータで,性別の度数分布表や,性別と教育年数の分割表を作成する。

with(d02, addmargins(table(sex)))
sex
  1   2   3 Sum 
382 368  50 800 
with(d02, addmargins(table(sex, school)))
     school
sex     9  12  14  16  18 Sum
  1    22 161   5 174  20 382
  2    39 110  46 157  16 368
  3     3   8   8  24   7  50
  Sum  64 279  59 355  43 800

SPSSの様な変数ラベルや値ラベルがついていないのでこれでは分りにくい。
性別の度数分布表をオブジェクトに格納して,変数値や変数にラベルを付けてみよう。

t10 <- with(d02, table(sex))
rownames(t10) <- c("1 男性", "2 女性", "3 その他")

t10
sex
  1 男性   2 女性 3 その他 
     382      368       50 
names(dimnames(t10)) <- "性別 sex"
t10
性別 sex
  1 男性   2 女性 3 その他 
     382      368       50 

 上と同様にして,分割表の行と列にラベルを付けよう。

分割表の行と列にラベルを付ける
t11 <- with(d02, table(sex, school))
rownames(t11) <- c("1 男性", "2 女性", "3 その他")
colnames(t11) <- c("1 中学", "2 高校", "3 短大", "4 四大", "5 院")
names(dimnames(t11)) <- c("性別 sex", "最終学歴 school")

t11
          最終学歴 school
性別 sex 1 中学 2 高校 3 短大 4 四大 5 院
  1 男性       22    161      5    174   20
  2 女性       39    110     46    157   16
  3 その他      3      8      8     24    7

 変数そのものにラベルを貼り付けるのとは違って集計表の行や列に名前を付けているだけなので,正直このやり方ではいちいち非常に面倒であり,SPSS的な変数ラベル,値ラベルを使いたいと思う事も多いだろう。
しかしそこは逆に,常に値とラベルの対応に注意させられる事で,ラベルに惑わされて実態としての値(数字)を取り違え,処理や解釈を誤ると云うリスクがなくなると前向きに考えておこう(実際初心者がSPSSを使う場合にこのリスクは小さくない)。
Rで変数ラベル,値ラベルを使用可能にする工夫も幾つか開発されているようが,ここでは極力追加アプリケイションや追加パッケイジをインストールしなくて済む方法を紹介しているので,割愛する。

2.4 因子型変数を用いてラベルを付ける

 分割表を作成するごとにそれの行や列にラベルを付ける上記のやり方は,同じ変数で沢山の分割表を作成する場合には効率的とは言えない。
 ここでは,数値型変数から,実質的にそれと同じ因子型変数を作り,それで分割表や(下記の)モザイクプロットを表示する方法を推奨する。

数値型変数は小文字の変数名を持っているので,大文字の変数名で対応する因子型変数を作成する方針とする。

d02$SEX <- factor(d02$sex, 
                  levels = 1:3, 
                  labels = c("male", "female", "others"))

新変数を作成したら必ず新旧変数の分割表で齟齬が無いかどうかを確認する。

with(d02, table(sex, SEX, useNA = "ifany"))
   SEX
sex male female others
  1  382      0      0
  2    0    368      0
  3    0      0     50
d02$SCHOOL <- factor(d02$school,
                     levels = c(9, 12, 14, 16, 18), 
                     labels = c("JH", "High", "College", "UNIV", "Graduate"))

with(d02, table(school, SCHOOL, useNA = "ifany"))
      SCHOOL
school  JH High College UNIV Graduate
    9   64    0       0    0        0
    12   0  279       0    0        0
    14   0    0      59    0        0
    16   0    0       0  355        0
    18   0    0       0    0       43

因子型変数で分割表を作成する。改めて値ラベルは付けない。

with(d02, table(SEX, SCHOOL, useNA = "ifany"))
        SCHOOL
SEX       JH High College UNIV Graduate
  male    22  161       5  174       20
  female  39  110      46  157       16
  others   3    8       8   24        7

変数のラベルはつかないが,分割表の行と列にラベルを付けなくても,因子型変数のラベルが用いられるので数値だけで表示されるより分かり易い。
同じ質問項目に数値型変数と因子型変数の二つを用意しておくと,場合に応じて使い分けられて便利かも知れない。

2.5 分割表をそのまま図示する

分割表をグラフ表示するには,伝統的な帯グラフよりはモザイク・プロットの方が良い。
そうでなければ横に並べた棒グラフだろう。

mosaicplot(t11, 
           col = terrain.colors(dim(t11)[2]))

barplot(t(t11), 
        beside = T, 
        col    = terrain.colors(dim(t11)[2]), 
        legend = T)

mosaicplot(t11, shade = T)

複雑なモザイクプロット

因子型変数を用い,オプションを色々と設定した。
最初は複数のオプションを一度に設定するのではなく,一つ一つ追加してそれがどの様な設定を行うのかを確認して理解してから使うこと。

with(d02, 
  mosaicplot(SEX ~ SCHOOL, 
             col    = terrain.colors(5),
             las    = 1, 
             off    = 3, 
             border = "#00000060",
             cex    = 1.0,  
             main   = "性別と学歴のモザイクプロット",
             sub    = "(模擬データによる)",
             xlab   = "性別", 
             ylab   = "学歴")
)