Écriture d'une fonction personnalisée pour convertir une classe de variables dans un dataframe basé sur une autre table
J'essaye d'écrire une fonction qui pourrait prendre:
- data frame (
df_1
) dont les classes des colonnes doivent être converties - une autre trame de données (
df_2
) qui a une ligne pour chaque variable dedf_1
- une colonne
df_2
qui spécifie la classe dans laquelle chaque variabledf_1
doit être convertie
Exemple
1 - Trame de données ( df_1
) avec mes données (et les classes de variables à convertir)
library(tibble)
library(dplyr)
set.seed(2021)
df_1 <-
tibble(name = c("john", "jack", "mary", "matt", "elizabeth", "richard", "carlos", "george", "ferdinand", "william"),
height = sample(155:200, size = 10),
weight = sample(50:100, size = 10),
age = sample(20:100, size = 10),
gender = sample(c("male", "female"), size = 10, replace = TRUE),
preferred_pet = sample(c("dog", "cat", "frog", "rabbit"), size= 10, replace = TRUE)) %>%
mutate(across(everything(), as.character))
## # A tibble: 10 x 6
## name height weight age gender preferred_pet
## <chr> <chr> <chr> <chr> <chr> <chr>
## 1 john 161 100 38 female frog
## 2 jack 192 67 87 female dog
## 3 mary 193 52 24 male rabbit
## 4 matt 166 95 92 male dog
## 5 elizabeth 160 89 82 female cat
## 6 richard 199 75 57 male dog
## 7 carlos 195 85 37 female rabbit
## 8 george 159 86 62 male rabbit
## 9 ferdinand 177 71 78 female cat
## 10 william 197 80 89 female rabbit
2 - Trame de données ( df_2
) avec classes pour convertir les df_1
colonnes en
set.seed(2021)
df_2 <-
tibble(var_name = c("name", "height", "weight", "gender", "preferred_pet", "record_creation"),
var_class = c("character", "numeric", "numeric", "factor", "factor", "datetime")) %>%
slice(sample(1:n()))
## # A tibble: 6 x 2
## var_name var_class
## <chr> <chr>
## 1 weight numeric
## 2 record_creation datetime
## 3 height numeric
## 4 name character
## 5 gender factor
## 6 preferred_pet factor
3 - Construire une fonction pour la conversion de classe
J'ai vu la solution de @ akrun ici , qui semble assez proche de ce que j'essaie de réaliser.
library(purrr)
library(stringr)
my_df <- iris
my_types <- c("factor", "character", "double", "logical", "character")
my_df[] <- map2(my_df, str_c("as.", my_types), ~ get(.y)(.x))
Cependant, cette solution ne traite pas des situations comme la mienne, où les noms de variables de df_1
n'apparaissent pas nécessairement dans df_2
, et de même, df_2$var_name
incluent des variables qui n'apparaissent pas nécessairement dans df_1
.
Je serai heureux de toute idée de construction d'une fonction pour convertir df_1
les classes vars de en fonction des informations trouvées dans df_2
. Trouver une solution à l'aide de tidyverse
fonctions serait idéal. Merci!
Réponses
Voici une approche exploitant across
et cur_column
:
library(dplyr) #version >= 1.0.0
df_1 %>%
mutate(across(any_of(df_2$var_name), ~get(paste0("as.",df_2[df_2$var_name == cur_column(),"var_class"]))(.x)))
# A tibble: 10 x 6
name height weight age gender preferred_pet
<chr> <dbl> <dbl> <chr> <fct> <fct>
1 john 161 100 38 female frog
2 jack 192 67 87 female dog
3 mary 193 52 24 male rabbit
4 matt 166 95 92 male dog
5 elizabeth 160 89 82 female cat
6 richard 199 75 57 male dog
7 carlos 195 85 37 female rabbit
8 george 159 86 62 male rabbit
9 ferdinand 177 71 78 female cat
10 william 197 80 89 female rabbit
L' any_of
assistant de sélection garantit que vous essayez uniquement de muter les colonnes présentes dans df_2
.
Le deuxième argument est la fonction qui est appliquée aux colonnes présentes. Vous pouvez utiliser cur_column()
pour avoir accès au nom de la colonne en cours de mutation. À partir de là, nous recherchons simplement ce nom de colonne df_2
et retournons le nom de var_class
votre choix. Ensuite, utilisez get()
from base R pour renvoyer la fonction appropriée et appliquez-la à la colonne avec (.x)
.
Si vous vouliez définir une fonction et passer les noms de colonnes sans guillemets comme vous le feriez avec d'autres fonctions tidyverse, vous pouvez utiliser rlang::enquo
:
library(rlang)
change_class_by_table <- function(data,data_ref,column_name,column_class){
data %>%
mutate(across(any_of(pull(data_ref,!!enquo(column_name))),
~get(paste0("as.",filter(data_ref, !!enquo(column_name) == cur_column()) %>%
pull(!!enquo(column_class))))(.x)))
}
change_class_by_table(df_1,df_2,var_name,var_class)
## A tibble: 10 x 6
# name height weight age gender preferred_pet
# <chr> <dbl> <dbl> <chr> <fct> <fct>
# 1 john 161 100 38 female frog
# 2 jack 192 67 87 female dog
# 3 mary 193 52 24 male rabbit
# ...