top of page

Data Exploration and Hypothesis Generation: Deep Dive into Categorical Data and Feature Engineering




In this tutorial, we will explore the vast world of categorical data, learn about feature engineering, and dive into the generation of hypotheses in the field of data science. Each section will be filled with practical explanations, illustrative examples, and plentiful code snippets to ensure a thorough understanding.


I. Considerations for Categorical Data


A. Purpose of Exploratory Data Analysis (EDA)


Exploratory Data Analysis (EDA) is a significant step in the data science pipeline. The objectives of EDA include:

  1. Detection of patterns and relationships: EDA is like a detective trying to find clues in data. It aims to find any discernible patterns, trends, or relationships that exist in the data.

  2. Generation of questions or hypotheses: EDA helps formulate insightful questions or hypotheses based on the patterns detected.

  3. Preparation of data for machine learning models: The insights from EDA guide us in preparing the data for further analysis, including building machine learning models.

Imagine we're trying to bake a cake. EDA would be like checking the quality and mix of the ingredients before starting to bake.

# Importing required libraries
import pandas as pd
import numpy as np

# Let's say we have a dataset of baking ingredients
df = pd.DataFrame({'Ingredient':['flour', 'sugar', 'eggs', 'butter', 'vanilla'],
                   'Quantity':[2, 1.5, 2, 0.5, 0.02],
                   'Quality':['Good', 'Average', 'Poor', 'Good', 'Average']})

print(df)

In this small example, a simple data overview might already prompt questions like: Will the quality of the eggs affect the outcome of the cake?


B. Representation of Data


1. Importance of data being representative of the population


When working with data, we want our dataset to be representative of the population we are interested in. If we are studying income levels in the USA, a sample that only includes Silicon Valley residents will not represent the entire population accurately.


2. Example of studying income levels in the USA


For instance, if we had a dataset of income levels in the USA, we would check if our sample is representative by comparing it with known demographics.

# Load the income dataset (fictional example)
income_data = pd.read_csv('income_data.csv')

# Check the distribution of the data
income_data['income'].hist()


After running this code, we should see a histogram of income levels, which we could then compare to known demographics to see if our sample is representative. If our data is skewed towards high-income individuals, it might not represent the entire country accurately.


C. Categorical Classes


1. Importance of representation of classes (or labels)


When dealing with categorical data, it is vital that our classes are representative. For example, if we're studying marital status, we need our data to have sufficient representation of all status classes - single, married, divorced, widowed, etc.


2. Example of marital status as classes


Let's consider an example with a data frame of individuals and their marital statuses.

# Creating a DataFrame
marital_data = pd.DataFrame({'Name':['Alice', 'Bob', 'Charlie', 'David', 'Eve'],
                             'Marital Status':['Single', 'Married', 'Divorced', 'Widowed', 'Single']})

# Count the instances of each marital status
marital_data['Marital Status'].value_counts()

Running this code would provide us with a count of each marital status in our dataset.


D. Class Imbalance


1. Explanation of class imbalance


Class imbalance occurs when the classes in our categorical data are not represented equally. This can lead to bias in our analysis or machine learning models.


2. Example of class imbalance in a study about marital status


Continuing with our marital status example, let's say our data is skewed towards single individuals.

# Creating an imbalanced DataFrame
imbalanced_data = pd.DataFrame({'Name':['Alice', 'Bob', 'Charlie', 'David', 'Eve'],
                                'Marital Status':['Single', 'Single', 'Single', 'Single', 'Married']})

# Count the instances of each marital status
imbalanced_data['Marital Status'].value_counts()


In this case, 'Single' significantly outnumbers 'Married', which represents a class imbalance issue.


3. Potential biases introduced by class imbalance


In our example, class imbalance could lead to inaccurate conclusions. If we're trying to draw insights about married people's habits from this data, our analysis will be biased due to the under-representation of married individuals.


E. Class and Relative Class Frequency


1. How to calculate the number of observations per class


The number of observations per class can be calculated with the value_counts() function in pandas.

# Count the instances of each marital status
marital_data['Marital Status'].value_counts()


2. Introduction to relative frequencies


Relative frequency provides us with a ratio that compares the count of a specific class to the total number of observations.


3. How to calculate relative frequencies


Relative frequencies can be calculated in pandas using the value_counts() function with the argument normalize=True.

# Calculate the relative frequency of each marital status
marital_data['Marital Status'].value_counts(normalize=True)


The output would be the relative frequency of each class, which can help identify any class imbalance.


F. Cross-tabulation


1. Explanation of cross-tabulation


Cross-tabulation is a technique used to examine the relationship between two or more categorical variables. It provides a basic picture of interrelation between variables and can help find interactions between them.


2. How to create a cross-tabulation using pandas


Pandas provide the crosstab() function to perform cross-tabulation.

# Add a new column for employment status
marital_data['Employment'] = ['Employed', 'Unemployed', 'Employed', 'Employed', 'Unemployed']

# Cross-tabulation of marital and employment status
pd.crosstab(marital_data['Marital Status'], marital_data['Employment'])


3. Use of cross-tabulation to examine frequency of combinations of classes


The resulting table shows the frequency of each combination of marital status and employment status, allowing us to see how these categorical variables interact with each other.


II. Feature Generation


A. Data Formatting and Its Limitations


The way data is formatted can greatly influence the results of our analysis. In some cases, raw data might not be in the best format for certain analytical tasks. Fortunately, we can overcome these limitations through feature generation.

Imagine if we have a bag of mixed fruits, and we need to sort them by type. Data formatting is like separating each type of fruit into different baskets. Feature generation is like creating new categories such as "tropical fruits", "local fruits", and so on, to enhance our understanding of the data.


B. Correlation Analysis


1. How to check correlation with a heatmap


Correlation analysis helps identify relationships between numerical variables. To demonstrate this, let's use the seaborn library to create a heatmap.

# Import required libraries
import seaborn as sns
import matplotlib.pyplot as plt

# Generate a random DataFrame
np.random.seed(0)
df = pd.DataFrame(np.random.rand(10, 5), columns=['A', 'B', 'C', 'D', 'E'])

# Calculate correlations
corr = df.corr()

# Plot a heatmap
plt.figure(figsize=(8, 6))
sns.heatmap(corr, annot=True)
plt.show()


After running the above code, you'll see a heatmap that shows the correlation between each pair of variables.


2. Handling data types in correlation analysis


Correlation analysis typically works with continuous numerical variables. For categorical data, we can convert categories into numerical form (like "one-hot encoding") or use techniques designed for categorical data, like chi-square test.


C. Cleaning Data for Analysis


1. Cleaning and converting data types for correlation analysis


Data cleaning is crucial for correlation analysis. This involves handling missing values, outliers, and converting data types as required. For instance, we may convert a categorical feature into numerical using one-hot encoding.


Let's consider an example where we clean up data for total stops in a transportation dataset:

# Assuming we have a DataFrame 'transport_data'
transport_data = pd.DataFrame({'Bus': ['A', 'B', 'C'],
                               'Total Stops': ['10', '20', 'Unknown']})

# Replace 'Unknown' with NaN
transport_data['Total Stops'] = transport_data['Total Stops'].replace('Unknown', np.nan)

# Convert the 'Total Stops' column to numeric
transport_data['Total Stops'] = pd.to_numeric(transport_data['Total Stops'])

print(transport_data)


This cleans up our total stops data and readies it for further analysis.


2. Example: Cleaning total stops data


Assume that we have a dataset where total stops data is given in the format "10 stops". Here's how we can clean it:

# Assume a DataFrame with messy data
messy_data = pd.DataFrame({'Bus': ['A', 'B', 'C'],
                           'Total Stops': ['10 stops', '20 stops', 'Unknown']})

# Strip ' stops' from 'Total Stops' and replace 'Unknown' with NaN
messy_data['Total Stops'] = messy_data['Total Stops'].str.rstrip(' stops').replace('Unknown', np.nan)

# Convert the 'Total Stops' column to numeric
messy_data['Total Stops'] = pd.to_numeric(messy_data['Total Stops'])

print(messy_data)


D. Date-Time Variables


1. Extracting attributes from date-time variables


Date-time variables are packed with useful information. We can extract different attributes such as the year, month, day, hour, minute, and second, or even the day of the week.


Here's how we can do it:

# Create a DataFrame with date-time data
date_data = pd.DataFrame({'Name': ['Alice', 'Bob', 'Charlie'],
                          'Timestamp': pd.to_datetime(['2022-01-01 10:30', '2022-02-02 14:45', '2022-03-03 20:00'])})

# Extract year, month, day, and hour
date_data['Year'] = date_data['Timestamp'].dt.year
date_data['Month'] = date_data['Timestamp'].dt.month
date_data['Day'] = date_data['Timestamp'].dt.day
date_data['Hour'] = date_data['Timestamp'].dt.hour

print(date_data)


2. How to create new columns in a DataFrame using these attributes


Creating new columns from these attributes can give us new features that could potentially improve our model's performance. In the above code, we have already created new columns for Year, Month, Day, and Hour from the Timestamp column.


3. Extracting the hour of departure and arrival times


If we have columns for departure and arrival times, we can extract meaningful features like travel duration. Here's an example:

# Assuming we have a DataFrame 'flight_data' with departure and arrival times
flight_data = pd.DataFrame({'Flight': ['A', 'B', 'C'],
                            'Departure': pd.to_datetime(['2022-01-01 10:30', '2022-02-02 14:45', '2022-03-03 20:00']),
                            'Arrival': pd.to_datetime(['2022-01-01 14:30', '2022-02-02 18:45', '2022-03-03 22:00'])})

# Calculate travel duration
flight_data['Duration'] = (flight_data['Arrival'] - flight_data['Departure']).dt.total_seconds() / 3600

print(flight_data)


In this code, we calculate the travel duration in hours and store it in a new column.


E. Creating Categories


1. Grouping numeric data and labeling them as classes


Numeric data can be grouped into categories to provide more descriptive analysis. We can group data into bins using pandas' cut() function.


2. Example: Creating flight categories based on price ranges


Let's say we have a dataset of flight prices, and we want to categorize the flights as 'cheap', 'medium', or 'expensive'.

# Assume a DataFrame with flight prices
flight_data['Price'] = [100, 200, 300]

# Define price ranges and labels
bins = [0, 150, 250, np.inf]
labels = ['Cheap', 'Medium', 'Expensive']

# Create a new column with price categories
flight_data['Price Category'] = pd.cut(flight_data['Price'], bins=bins, labels=labels)

print(flight_data)


F. Descriptive Statistics


1. Use of quartiles to split data across a price range


Descriptive statistics like quartiles can also be used to create categories. For example, we can split data into quartiles and label them as 'Q1', 'Q2', 'Q3', and 'Q4'.

# Create a new column with quartile categories
flight_data['Price Quartile'] = pd.qcut(flight_data['Price'], q=4, labels=['Q1', 'Q2', 'Q3', 'Q4'])

print(flight_data)


2. Storing percentiles and maximum values as new variables


We can also store percentiles and maximum values as new variables to provide additional insights.

# Calculate and store the 90th percentile and maximum price
flight_data['90th Percentile Price'] = flight_data['Price'].quantile(0.9)
flight_data['Max Price'] = flight_data['Price'].max()

print(flight_data)


The new variables give us information about the overall price distribution.

With the foundation of feature generation firmly in place, the next stage of our tutorial is about hypothesis generation.


III. Hypothesis Generation


A. Importance of Hypothesis Generation


Hypothesis generation plays a crucial role in the data analysis pipeline. It guides the direction of our analysis and helps us build a story around our data. Think of it as forming a theory that you aim to prove or disprove with the data at hand.


B. How to Generate Hypotheses


1. Use domain knowledge and initial exploration


Hypotheses can be generated based on domain knowledge, initial data exploration, or a combination of both. It's about making educated guesses about what patterns or relationships might exist in the data.


For example, if we're working with a dataset about marathon runners, we might hypothesize that "runners with a lower body mass index (BMI) will have faster finish times".


2. Make hypotheses testable


It's important that our hypotheses are testable. A testable hypothesis allows us to use statistical tests to confirm or reject it.

Our marathon runners hypothesis is testable because we can group runners by BMI and compare their finish times.


C. Testing Hypotheses


1. Use statistical tests


Hypotheses can be tested using various statistical tests, depending on the nature of the data and the hypothesis. For example, to test our marathon runners hypothesis, we might use a t-test to compare the mean finish times of two groups.


2. Correlation analysis for testing hypotheses


A correlation analysis can be used to test hypotheses about relationships between variables. For instance, we could calculate the correlation between BMI and finish time in our marathon runners data.


D. Generating New Features Based on Hypotheses


1. Example: Creating a new feature for BMI categories


New features can be created based on our hypotheses. If we hypothesize that BMI affects finish times, we might create a new feature that categorizes runners into 'low', 'medium', and 'high' BMI.

# Assume we have a DataFrame 'runners_data' with BMI data
runners_data = pd.DataFrame({'Runner': ['Alice', 'Bob', 'Charlie'],
                             'BMI': [18.5, 24.9, 30.0]})

# Define BMI categories
bins = [0, 18.5, 24.9, np.inf]
labels = ['Low', 'Medium', 'High']

# Create a new column for BMI categories
runners_data['BMI Category'] = pd.cut(runners_data['BMI'], bins=bins, labels=labels)

print(runners_data)


2. Use new features to test hypotheses


The newly created features can be used to test our hypotheses. We might compare the finish times across the BMI categories to test our hypothesis.

That's all for this tutorial. We've explored categorical data, delved into feature engineering, and looked at hypothesis generation. Each section provided practical examples to support the concepts discussed. Keep exploring and happy data science journey!

bottom of page