Пользовательская сгруппированная функция dplyr (sample_n) - PullRequest
3 голосов
/ 14 апреля 2019

Я пытаюсь применить функцию выборки сгруппированным образом к фрейму данных, где он должен выбрать n выборок из каждой группы, или всех членов группы, если размер группы меньше, чем n .

Используя dplyr, я сначала попробовал

library(dplyr)
mtcars %>% group_by(cyl) %>% sample_n(2)

. Это работает, когда n меньше всех размеров группы, но не принимает полную группу, когда я выбираю n больше, чем размер группы (обратите внимание, чтов одной из групп цилиндров есть 7 машин):

mtcars %>% group_by(cyl) %>% sample_n(8)
Error: `size` must be less or equal than 7 (size of data), 
set `replace` = TRUE to use sampling with replacement

Я попытался решить эту проблему, создав адаптированную функцию group_n, например:

sample_n_or_all <- function(tbl, n) {
  if (nrow(tbl) < n)return(tbl)
  sample_n(tbl, n)
}

, но с использованием своей пользовательской функции (mtcars %>% group_by(cyl) %>% sample_n_or_all(8)) генерирует ту же ошибку.

Какие-либо предложения, как я могу адаптировать свою функцию, чтобы я мог применить ее к каждой из групп?Или другое решение проблемы?

Ответы [ 2 ]

2 голосов
/ 14 апреля 2019

Мы могли бы проверить количество строк в группе и передать значение в sample_n соответственно.

library(dplyr)
n <- 8

temp <- mtcars %>% group_by(cyl) %>% sample_n(if(n() < n) n() else n) 
temp

#    mpg   cyl  disp    hp  drat    wt  qsec    vs    am  gear  carb
#   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
# 1  21.4     4 121     109  4.11  2.78  18.6     1     1     4     2
# 2  27.3     4  79      66  4.08  1.94  18.9     1     1     4     1
# 3  24.4     4 147.     62  3.69  3.19  20       1     0     4     2
# 4  22.8     4 108      93  3.85  2.32  18.6     1     1     4     1
# 5  26       4 120.     91  4.43  2.14  16.7     0     1     5     2
# 6  33.9     4  71.1    65  4.22  1.84  19.9     1     1     4     1
# 7  30.4     4  75.7    52  4.93  1.62  18.5     1     1     4     2
# 8  30.4     4  95.1   113  3.77  1.51  16.9     1     1     5     2
# 9  21       6 160     110  3.9   2.62  16.5     0     1     4     4
#10  17.8     6 168.    123  3.92  3.44  18.9     1     0     4     4
# … with 13 more rows

После этого мы можем проверить количество строк в каждой группе.

table(temp$cyl)

#4 6 8 
#8 7 8 

table(mtcars$cyl)

# 4  6  8 
#11  7 14 
1 голос
/ 14 апреля 2019

Мы можем сделать это без использования логического условия с pmin

library(dplyr)
tmp <- mtcars %>%
         group_by(cyl) %>%
         sample_n(pmin(n(), n))
# A tibble: 23 x 11
# Groups:   cyl [3]
#     mpg   cyl  disp    hp  drat    wt  qsec    vs    am  gear  carb
#   <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
# 1  33.9     4  71.1    65  4.22  1.84  19.9     1     1     4     1
# 2  27.3     4  79      66  4.08  1.94  18.9     1     1     4     1
# 3  21.4     4 121     109  4.11  2.78  18.6     1     1     4     2
# 4  30.4     4  75.7    52  4.93  1.62  18.5     1     1     4     2
# 5  21.5     4 120.     97  3.7   2.46  20.0     1     0     3     1
# 6  32.4     4  78.7    66  4.08  2.2   19.5     1     1     4     1
# 7  30.4     4  95.1   113  3.77  1.51  16.9     1     1     5     2
# 8  26       4 120.     91  4.43  2.14  16.7     0     1     5     2
# 9  17.8     6 168.    123  3.92  3.44  18.9     1     0     4     4
#10  21       6 160     110  3.9   2.62  16.5     0     1     4     4
# … with 13 more rows

-checking

table(tmp$cyl)
# 4 6 8 
# 8 7 8 
...