Encoding: from a dataframe to a numerical matrix for machine learning#

This example shows how to transform a rich dataframe with columns of various types into a numerical matrix on which machine-learning algorithms can be applied. We study the case of predicting wages using the employee salaries dataset.

Easy learning on a dataframe#

Let’s first retrieve the dataset, using one of the downloaders from the skrub.datasets module. As all the downloaders, fetch_employee_salaries() returns a dataset with attributes X, and y. X is a dataframe which contains the features (aka design matrix, explanatory variables, independent variables). y is a column (pandas Series) which contains the target (aka dependent, response variable) that we want to learn to predict from X. In this case y is the annual salary.

gender department department_name division assignment_category employee_position_title date_first_hired year_first_hired
0 F POL Department of Police MSB Information Mgmt and Tech Division Records... Fulltime-Regular Office Services Coordinator 09/22/1986 1986
1 M POL Department of Police ISB Major Crimes Division Fugitive Section Fulltime-Regular Master Police Officer 09/12/1988 1988
2 F HHS Department of Health and Human Services Adult Protective and Case Management Services Fulltime-Regular Social Worker IV 11/19/1989 1989
3 M COR Correction and Rehabilitation PRRS Facility and Security Fulltime-Regular Resident Supervisor II 05/05/2014 2014
4 M HCA Department of Housing and Community Affairs Affordable Housing Programs Fulltime-Regular Planning Specialist III 03/05/2007 2007
... ... ... ... ... ... ... ... ...
9223 F HHS Department of Health and Human Services School Based Health Centers Fulltime-Regular Community Health Nurse II 11/03/2015 2015
9224 F FRS Fire and Rescue Services Human Resources Division Fulltime-Regular Fire/Rescue Division Chief 11/28/1988 1988
9225 M HHS Department of Health and Human Services Child and Adolescent Mental Health Clinic Serv... Parttime-Regular Medical Doctor IV - Psychiatrist 04/30/2001 2001
9226 M CCL County Council Council Central Staff Fulltime-Regular Manager II 09/05/2006 2006
9227 M DLC Department of Liquor Control Licensure, Regulation and Education Fulltime-Regular Alcohol/Tobacco Enforcement Specialist II 01/30/2012 2012

9228 rows × 8 columns



Most machine-learning algorithms work with arrays of numbers. The challenge here is that the employees dataframe is a heterogeneous set of columns: some are numerical ('year_first_hired'), some dates ('date_first_hired'), some have a few categorical entries ('gender'), some many ('employee_position_title'). Therefore our table needs to be “vectorized”: processed to extract numeric features.

skrub provides an easy way to build a simple but reliable machine-learning model which includes this step, working well on most tabular data.

from sklearn.model_selection import cross_validate

from skrub import tabular_learner

model = tabular_learner("regressor")
results = cross_validate(model, employees, salaries)
results["test_score"]
array([0.89370447, 0.89279068, 0.92282557, 0.92319094, 0.92162666])

The estimator returned by tabular_learner combines 2 steps:

Pipeline(steps=[('tablevectorizer',
                 TableVectorizer(high_cardinality=MinHashEncoder(),
                                 low_cardinality=ToCategorical())),
                ('histgradientboostingregressor',
                 HistGradientBoostingRegressor(categorical_features='from_dtype'))])
In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.


In the rest of this example, we focus on the first step and explore the capabilities of skrub’s TableVectorizer.


More details on encoding tabular data#

gender_F gender_M gender_nan department_BOA department_BOE department_CAT department_CCL department_CEC department_CEX department_COR department_CUS department_DEP department_DGS department_DHS department_DLC department_DOT department_DPS department_DTS department_ECM department_FIN department_FRS department_HCA department_HHS department_HRC department_IGR department_LIB department_MPB department_NDA department_OAG department_OCP department_OHR department_OIG department_OLO department_OMB department_PIO department_POL department_PRO department_REC department_SHF department_ZAH ... division: 3rd, 1st, 2nd division: planning, building, zoning division: maintenance, facilities, facility division: investigative, investigations, criminal assignment_category_Parttime-Regular employee_position_title: government, intergovernmental, partnerships employee_position_title: manager, budget, management employee_position_title: officer, office, police employee_position_title: operator, equipment, bus employee_position_title: master, registered, water employee_position_title: income, assistance, client employee_position_title: school, room, pool employee_position_title: administrative, administration, administrator employee_position_title: firefighter, recruit, rescuer employee_position_title: information, recreation, technology employee_position_title: candidate, librarian, library employee_position_title: correctional, correction, corporal employee_position_title: enforcement, permitting, inspector employee_position_title: community, security, health employee_position_title: communications, telecommunications, safety employee_position_title: coordinator, coordinating, forest employee_position_title: legislative, principal, executive employee_position_title: lieutenant, shift, commander employee_position_title: liquor, clerk, store employee_position_title: warehouse, craftsworker, welfare employee_position_title: specialist, special, resource employee_position_title: sergeant, supervisor, supervisory employee_position_title: services, service, customer employee_position_title: program, programs, projects employee_position_title: crossing, purchasing, background employee_position_title: captain, chief, battalion employee_position_title: transit, supply, systems employee_position_title: therapist, sheriff, deputy employee_position_title: accountant, assistant, county employee_position_title: technician, mechanic, hvac date_first_hired_year date_first_hired_month date_first_hired_day date_first_hired_total_seconds year_first_hired
0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 ... 0.084437 0.101386 0.142456 0.184494 0.0 0.052376 0.053432 0.096772 0.065107 0.055215 0.053814 0.051793 0.053785 0.054604 0.052815 0.051348 0.061221 0.054183 0.052407 0.054309 37.314178 0.052686 0.053559 0.058823 0.053314 0.051221 0.077430 0.106833 0.052318 0.052643 0.055566 0.051594 0.053451 0.051464 0.051740 1986.0 9.0 22.0 5.277312e+08 1986.0
1 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 ... 0.057303 0.061263 0.298869 1.357173 0.0 0.055372 0.066821 21.653015 0.053874 6.721349 0.072347 0.052676 0.058768 0.059497 0.056476 0.059989 0.061188 0.051673 0.050752 0.053874 0.064682 0.053894 0.055396 0.059883 0.053780 0.056649 0.072889 0.059670 0.061136 0.057584 0.051865 0.054742 0.055348 0.061212 0.053599 1988.0 9.0 12.0 5.900256e+08 1988.0
2 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 ... 0.062049 0.103662 0.063493 0.065172 0.0 0.052249 0.069892 0.060906 0.053241 0.054522 0.057216 0.058755 0.066645 0.058907 0.059515 0.055525 0.064715 0.055650 0.052979 0.053271 0.052294 0.058338 0.050284 0.057747 20.846649 0.078081 0.052506 0.052298 0.058711 0.050144 0.050245 0.055302 0.053305 0.051801 0.058310 1989.0 11.0 19.0 6.274368e+08 1989.0
3 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 ... 0.056522 0.107372 12.863601 0.053880 0.0 14.925659 1.419785 0.097812 0.076152 0.081719 0.067563 0.051721 0.072268 0.207881 0.069593 0.086336 0.076538 2.715463 0.056677 0.063208 0.081949 0.096678 0.075276 0.084167 0.094277 0.083509 6.014322 4.363964 0.068992 0.055942 0.084979 0.085335 0.098350 0.082813 0.061071 2014.0 5.0 5.0 1.399248e+09 2014.0
4 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 ... 0.134079 0.190719 0.054453 0.061153 0.0 0.053179 0.978388 0.078697 0.051872 0.051008 0.098219 0.067489 0.373004 0.066388 0.082298 0.059011 0.062735 0.073210 0.064073 0.070602 0.051846 0.064031 0.059434 0.055886 0.064949 18.502352 0.061677 0.068452 0.060259 11.338292 0.051271 0.099370 0.079216 0.067208 0.145588 2007.0 3.0 5.0 1.173053e+09 2007.0
... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ...
9223 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 ... 0.074884 0.084751 0.065329 0.052372 0.0 0.050335 0.059221 0.055179 0.050001 0.050002 0.054487 0.066816 0.054456 0.053104 0.052261 0.051519 0.054379 0.053437 34.427555 0.060836 0.052113 0.052030 0.051304 0.052297 0.054026 0.053699 0.052872 0.066228 0.051434 0.050364 0.051139 0.052162 0.058601 0.052642 0.055502 2015.0 11.0 3.0 1.446509e+09 2015.0
9224 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 ... 0.058146 0.061356 0.070079 0.096831 0.0 0.547675 0.050008 0.051823 0.050003 0.072485 0.107820 0.058841 0.278764 0.078547 0.231580 1.483467 0.074705 0.050494 0.052028 0.081075 0.293937 0.460102 0.084112 0.052262 0.078958 0.071744 0.362470 0.334730 0.050004 0.108436 31.558735 0.069592 0.065757 0.576005 0.063842 1988.0 11.0 28.0 5.966784e+08 1988.0
9225 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 ... 0.060231 0.086235 0.073632 0.063962 1.0 37.809582 0.084200 0.106033 0.347169 0.054102 0.084662 0.088775 0.564686 0.061517 0.091966 0.220216 0.119530 1.974016 0.059164 0.098351 0.148773 0.267184 0.054260 0.132217 3.334373 0.143859 0.057950 0.060973 0.063373 0.052029 0.069464 0.077712 0.069251 0.087726 0.116889 2001.0 4.0 30.0 9.885888e+08 2001.0
9226 0.0 1.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 ... 0.056823 0.053412 0.062648 0.054477 0.0 0.051177 11.890217 0.063664 0.052104 0.056488 0.051602 0.051277 0.051735 0.061391 0.051577 0.051258 0.056622 0.055183 0.051899 0.050596 0.051728 0.050015 0.052891 0.053758 0.059854 0.051751 0.054678 0.053539 0.111496 0.050145 0.050002 0.051682 0.055092 0.051810 0.054769 2006.0 9.0 5.0 1.157414e+09 2006.0
9227 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 ... 0.055177 0.198250 0.080944 0.088362 0.0 0.066805 0.060020 0.178108 0.054979 0.051121 0.093041 0.109046 0.130590 0.052149 0.116539 0.061450 0.063039 34.218151 0.058894 0.065398 0.060806 0.054655 0.053361 0.058859 0.061970 16.401445 0.109994 0.056686 0.050883 7.270532 0.053687 0.055611 0.061840 0.209277 0.061065 2012.0 1.0 30.0 1.327882e+09 2012.0

9228 rows × 143 columns



From our 8 columns, the TableVectorizer has extracted 143 numerical features. Most of them are one-hot encoded representations of the categorical features. For example, we can see that 3 columns 'gender_F', 'gender_M', 'gender_nan' were created to encode the 'gender' column.

By performing appropriate transformations on our complex data, the TableVectorizer produced numeric features that we can use for machine-learning:

HistGradientBoostingRegressor()
In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.


The TableVectorizer bridges the gap between tabular data and machine-learning pipelines. It allows us to apply a machine-learning estimator to our dataframe without manual data wrangling and feature extraction.

Inspecting the TableVectorizer#

The TableVectorizer distinguishes between 4 basic kinds of columns (more may be added in the future). For each kind, it applies a different transformation, which we can configure. The kinds of columns and the default transformation for each of them are:

  • numeric columns: simply casting to floating-point

  • datetime columns: extracting features such as year, day, hour with the DatetimeEncoder

  • low-cardinality categorical columns: one-hot encoding

  • high-cardinality categorical columns: a simple and effective text representation pipeline provided by the GapEncoder

TableVectorizer()
In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.


We can inspect which transformation was chosen for a each column and retrieve the fitted transformer. vectorizer.kind_to_columns_ provides an overview of how the vectorizer categorized columns in our input:

{'numeric': ['year_first_hired'], 'datetime': ['date_first_hired'], 'low_cardinality': ['gender', 'department', 'department_name', 'assignment_category'], 'high_cardinality': ['division', 'employee_position_title'], 'specific': []}

The reverse mapping is given by:

{'year_first_hired': 'numeric', 'date_first_hired': 'datetime', 'gender': 'low_cardinality', 'department': 'low_cardinality', 'department_name': 'low_cardinality', 'assignment_category': 'low_cardinality', 'division': 'high_cardinality', 'employee_position_title': 'high_cardinality'}

vectorizer.transformers_ gives us a dictionary which maps column names to the corresponding transformer.

vectorizer.transformers_["date_first_hired"]
DatetimeEncoder()
In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.


We can also see which features in the vectorizer’s output were derived from a given input column.

vectorizer.input_to_outputs_["date_first_hired"]
['date_first_hired_year', 'date_first_hired_month', 'date_first_hired_day', 'date_first_hired_total_seconds']
date_first_hired_year date_first_hired_month date_first_hired_day date_first_hired_total_seconds
0 1986.0 9.0 22.0 5.277312e+08
1 1988.0 9.0 12.0 5.900256e+08
2 1989.0 11.0 19.0 6.274368e+08
3 2014.0 5.0 5.0 1.399248e+09
4 2007.0 3.0 5.0 1.173053e+09
... ... ... ... ...
9223 2015.0 11.0 3.0 1.446509e+09
9224 1988.0 11.0 28.0 5.966784e+08
9225 2001.0 4.0 30.0 9.885888e+08
9226 2006.0 9.0 5.0 1.157414e+09
9227 2012.0 1.0 30.0 1.327882e+09

9228 rows × 4 columns



Finally, we can go in the opposite direction: given a column in the input, find out from which input column it was derived.

vectorizer.output_to_input_["department_BOA"]
'department'

Dataframe preprocessing#

Note that "date_first_hired" has been recognized and processed as a datetime ßcolumn.

vectorizer.column_to_kind_["date_first_hired"]
'datetime'

But looking closer at our original dataframe, it was encoded as a string.

employees["date_first_hired"]
0       09/22/1986
1       09/12/1988
2       11/19/1989
3       05/05/2014
4       03/05/2007
           ...
9223    11/03/2015
9224    11/28/1988
9225    04/30/2001
9226    09/05/2006
9227    01/30/2012
Name: date_first_hired, Length: 9228, dtype: object

Note the dtype: object in the output above. Before applying the transformers we specify, the TableVectorizer performs a few preprocessing steps.

For example, strings commonly used to represent missing values such as "N/A" are replaced with actual null. As we saw above, columns containing strings that represent dates (e.g. '2024-05-15') are detected and converted to proper datetimes.

We can inspect the list of steps that were applied to a given column:

[CleanNullStrings(), DropIfTooManyNulls(), ToDatetime(), DatetimeEncoder(), {'date_first_hired_day': ToFloat32(), 'date_first_hired_month': ToFloat32(), ...}]

These preprocessing steps depend on the column:

[CleanNullStrings(), DropIfTooManyNulls(), ToStr(), OneHotEncoder(drop='if_binary', dtype='float32', handle_unknown='ignore',
              sparse_output=False), {'department_BOA': ToFloat32(), 'department_BOE': ToFloat32(), ...}]

A simple Pipeline for tabular data#

The TableVectorizer outputs data that can be understood by a scikit-learn estimator. Therefore we can easily build a 2-step scikit-learn Pipeline that we can fit, test or cross-validate and that works well on tabular data.

import numpy as np
from sklearn.ensemble import HistGradientBoostingRegressor
from sklearn.model_selection import cross_validate
from sklearn.pipeline import make_pipeline

pipeline = make_pipeline(TableVectorizer(), HistGradientBoostingRegressor())

results = cross_validate(pipeline, employees, salaries)
scores = results["test_score"]
print(f"R2 score:  mean: {np.mean(scores):.3f}; std: {np.std(scores):.3f}")
print(f"mean fit time: {np.mean(results['fit_time']):.3f} seconds")
R2 score:  mean: 0.920; std: 0.012
mean fit time: 8.035 seconds

Specializing the TableVectorizer for HistGradientBoosting#

The encoders used by default by the TableVectorizer are safe choices for a wide range of downstream estimators. If we know we want to use it with a HistGradientBoostingRegressor (or classifier) model, we can make some different choices that are only well-suited for tree-based models but can yield a faster pipeline. We make 2 changes.

The HistGradientBoostingRegressor has built-in support for categorical features, so we do not need to one-hot encode them. We do need to tell it which features should be treated as categorical with the categorical_features parameter. In recent versions of scikit-learn, we can set categorical_features='from_dtype', and it will treat all columns in the input that have a Categorical dtype as such. Therefore we change the encoder for low-cardinality columns: instead of OneHotEncoder, we use skrub’s ToCategorical. This transformer will simply ensure our columns have an actual Categorical dtype (as opposed to string for example), so that they can be recognized by the HistGradientBoostingRegressor.

The second change replaces the GapEncoder with a MinHashEncoder. The GapEncoder is a topic model. It produces interpretable embeddings in a vector space where distances are meaningful, which is great for interpretation and necessary for some downstream supervised learners such as linear models. However fitting the topic model is costly in computation time and memory. The MinHashEncoder produces features that are not easy to interpret, but that decision trees can efficiently use to test for the occurrence of particular character n-grams (more details are provided in its documentation). Therefore it can be a faster and very effective alternative, when the supervised learner is built on top of decision trees, which is the case for the HistGradientBoostingRegressor.

The resulting pipeline is identical to the one produced by default by tabular_learner.

from skrub import MinHashEncoder, ToCategorical

vectorizer = TableVectorizer(
    low_cardinality=ToCategorical(), high_cardinality=MinHashEncoder()
)
pipeline = make_pipeline(
    vectorizer, HistGradientBoostingRegressor(categorical_features="from_dtype")
)

results = cross_validate(pipeline, employees, salaries)
scores = results["test_score"]
print(f"R2 score:  mean: {np.mean(scores):.3f}; std: {np.std(scores):.3f}")
print(f"mean fit time: {np.mean(results['fit_time']):.3f} seconds")
R2 score:  mean: 0.911; std: 0.014
mean fit time: 1.342 seconds

We can see that this new pipeline achieves a similar score but is fitted much faster. This is mostly due to replacing GapEncoder with MinHashEncoder (however this makes the features less interpretable).

Feature importances in the statistical model#

As we just saw, we can fit a MinHashEncoder faster than a GapEncoder. However, the GapEncoder has a crucial advantage: each dimension of its output space is associated with a topic which can be inspected and interpreted. In this section, after training a regressor, we will plot the feature importances.

First, we train another scikit-learn regressor, the RandomForestRegressor:

from sklearn.ensemble import RandomForestRegressor

vectorizer = TableVectorizer()  # now using the default GapEncoder
regressor = RandomForestRegressor(n_estimators=50, max_depth=20, random_state=0)

pipeline = make_pipeline(vectorizer, regressor)
pipeline.fit(employees, salaries)
Pipeline(steps=[('tablevectorizer', TableVectorizer()),
                ('randomforestregressor',
                 RandomForestRegressor(max_depth=20, n_estimators=50,
                                       random_state=0))])
In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.


We are retrieving the feature importances:

And plotting the results:

import matplotlib.pyplot as plt

top_indices = indices[:20]
labels = vectorizer.get_feature_names_out()[top_indices]

plt.figure(figsize=(12, 9))
plt.barh(
    y=labels,
    width=avg_importances[top_indices],
    yerr=std_importances[top_indices],
    color="b",
)
plt.yticks(fontsize=15)
plt.title("Feature importances")
plt.tight_layout(pad=1)
plt.show()
Feature importances

The GapEncoder creates feature names that show the first 3 most important words in the topic associated with each feature. As we can see in the plot above, this helps inspecting the model. If we had used a MinHashEncoder instead, the features would be much less helpful, with names such as employee_position_title_0, employee_position_title_1, etc.

We can see that features such the time elapsed since being hired, having a full-time employment, and the position, seem to be the most informative for prediction. However, feature importances must not be over-interpreted – they capture statistical associations rather than causal effects. Moreover, the fast feature importance method used here suffers from biases favouring features with larger cardinality, as illustrated in a scikit-learn example. In general we should prefer permutation_importance(), but it is a slower method.

Conclusion#

In this example, we motivated the need for a simple machine learning pipeline, which we built using the TableVectorizer and a HistGradientBoostingRegressor.

We saw that by default, it works well on a heterogeneous dataset.

To better understand our dataset, and without much effort, we were also able to plot the feature importances.

Total running time of the script: (1 minutes 36.794 seconds)

Gallery generated by Sphinx-Gallery