Polars - The Better Pandas?

Contents

Introduction

Pandas is one of Python’s most used data manipulating libraries. It’s used widely but it has the problem that as soon as the data grows too big, it’s just too slow and companies use other solutions for data processing. Polars on the other hand is written in Rust and has an excellent Python library. In this blog post, we will explore the differences between the two, what kind of performance you can expect with either of them and which I recommend for what situations.

Performance

Let’s start with what most people are interested in: performance. For that we will test a couple of scenarios and compare their speeds. We’ll keep a running tab on the score. Let’s first create a dummy dataset with 10,000,000 rows and 21 columns (the first one being the ID).

import polars as pl
import pandas as pd
import numpy as np
import time
from loguru import logger
from functools import partial
import csv
import os

DATA_PATH = "data.csv"

def log_time(pl_time, pd_time):
logger.info(
f"Polars was {pd_time / pl_time}x {'faster' if pl_time < pd_time else 'slower'} than Pandas"
)

def create_csv():
n_rows = 10_000_000
n_cols = 20

ids = np.arange(n_rows, dtype="int")
cols = np.random.uniform(size=(n_rows, n_cols))

data = np.c_[ids, cols]
np.savetxt(DATA_PATH, data, delimiter=",")

def add_header_to_csv(input_filename, output_filename, headers):
    with open(input_filename, "r", newline="") as infile, open(
        output_filename, "w", newline=""
    ) as outfile:
        reader = csv.reader(infile)
        writer = csv.writer(outfile)
        writer.writerow(headers)
        for row in reader:
            writer.writerow(row)

headers = ["ID"] + [f"Col_{i}" for i in range(1, 21)]
add_header_to_csv(DATA_PATH, "data_with_headers.csv", headers)
os.remove(DATA_PATH)
os.rename("data_with_headers.csv", DATA_PATH)

This will create a ~5 GB file, so make sure you have enough space.

Read CSV

Now it’s time to test the performance. First, let’s check how long it takes to run px.read_csv (px is the placeholder for when I’m referring to both Pandas and Polars).

def test_open_csv():
    pl_start_time = time.time()
    pl.read_csv(DATA_PATH)
    pl_end_time = time.time()
    pl_time = pl_end_time - pl_start_time
    logger.info(f"Polars: {pl_time}")

    pd_start_time = time.time()
    pd.read_csv(DATA_PATH)
    pd_end_time = time.time()
    pd_time = pd_end_time - pd_start_time
    logger.info(f"Pandas: {pd_time}")

    log_time(pl_time, pd_time)
TaskPolarsPandas
read_csv2.926s21.093s

Polars was 7.2077x faster than Pandas.

Get min/max

def test_get_min():
    pl_df = pl.read_csv(DATA_PATH)
    pd_df = pd.read_csv(DATA_PATH)

    pl_val, pl_start_time, pl_end_time, pl_time = measure_time(pl_df.min, "Polars")
    pd_val, pd_start_time, pd_end_time, pd_time = measure_time(pd_df.min, "Pandas")

    assert np.allclose(pl_val.to_numpy(), pd_val.to_numpy())
    log_time(pl_time, pd_time)

def test_get_max():
    pl_df = pl.read_csv(DATA_PATH)
    pd_df = pd.read_csv(DATA_PATH)

    pl_val, pl_start_time, pl_end_time, pl_time = measure_time(pl_df.max, "Polars")
    pd_val, pd_start_time, pd_end_time, pd_time = measure_time(pd_df.max, "Pandas")

    assert np.allclose(pl_val.to_numpy(), pd_val.to_numpy())
    log_time(pl_time, pd_time)


TaskPolarsPandas
min0.1736s0.3275s
max0.1603s0.0.28s

Polars was 1.88691x faster than Pandas (min).

Polars was 1.72244x faster than Pandas (max).

Filtering

def test_filter():
    pl_df = pl.read_csv(DATA_PATH)
    pd_df = pd.read_csv(DATA_PATH)

    pd_fun = lambda df: df.loc[(df.iloc[:, 1:] > 0.5).all(axis=1)]
    pd_fun = partial(pd_fun, df=pd_df)
    pd_val, pd_start_time, pd_end_time, pd_time = measure_time(pd_fun, "Pandas")

    pl_fun = lambda df: df.filter(pl.all_horizontal(pl.exclude("ID") > 0.5))
    pl_fun = partial(pl_fun, df=pl_df)
    pl_val, pl_start_time, pl_end_time, pl_time = measure_time(pl_fun, "Polars")

    assert np.allclose(pd_val.to_numpy(), pl_val.to_numpy())
    log_time(pl_time, pd_time)

TaskPolarsPandas
filter0.1614s0.1860s

Polars was 1.1519x faster than Pandas.

Conclusion

From these tests, it’s clear that Polars is the better performer. Especially, the reading of the csv file is much faster. But performance isn’t the only metric you should look at when you’re checking out a new library or framework. It’s also important to check if its API is well designed enough, to allow you to solve your problems.

The API

It is generally true that Pandas looks and feels more like writing “normal” Python code and Polars kind of doesn’t. That’s because Polars developed a DSL - a domain specific language. It sets some expectations on the user on how Polars code should be structured. We already saw this in our little benchmark before:

# Pandas
df.loc[(df.iloc[:, 1:] > 0.5).all(axis=1)]

# Polars
df.filter(pl.all_horizontal(pl.exclude("ID") > 0.5))

As you can see, Polars is more “talkative”. In Pandas you use a lot of the standard Python notation, like slicing, whereas in Polars you describe more what you want. In Polars, there are 2 main concepts:

  1. Context
  2. Expressions

And in the code you would often write something like:

[CONTEXT_DF].[CONTEXT_METHOD](EXPRESSION)

The filter function is one of those context methods (along with select and with_columns).

Let’s look at some other examples where the API of Pandas and Polars differ. First, let’s explore the basics, e.g. how would I get rid of a column.

In Pandas, you would use slicing with the iloc property, whereas in Polars, you use the select method along with the exclude expression:

df_pandas_sliced = df_pandas.iloc[:, 1:]
df_polars_sliced = df_polars.select(pl.exclude("ID"))
# or df_polars_sliced = df_polars.drop(df_polars.columns[0])
# if you don't know the column name
# That would also work for Pandas

Which one do you prefer? For me, it’s the Polars API. Let’s look at another example: normalization.

In Polars, you would write it like this:

age_min = df.select(pl.col("Age").min()).to_numpy()[0, 0]
age_max = df.select(pl.col("Age").max()).to_numpy()[0, 0]
df = df.select(
    ((pl.col("Age") - age_min) / (age_max - age_min)).alias("Age"),
    pl.exclude("Age"),
)

And in Pandas, it’s this:

age_min = df["Age"].min()
age_max = df["Age"].max()

df["Age"] = (df["Age"] - age_min) / (age_max - age_min)

This point clearly goes to Pandas.

(At this point, it needs to be stated that I’m by no means an mega expert on both of these libraries. Especially in Polars, which is what I’m in the middle of learning - so perhaps there is a better way to handle normalization than what I conjured up)

What this means for you

Polars is objectively the faster library compared to Pandas and if that matters to you, then Polars is your go-to library. But beware, that once your data gets in the terrabyte realm, even Polars themselves recommend you choose a different tool.

Pandas is a good choice for you, if you enjoy having tons of examples already out there. In terms of documentation, I’d say both are very well documented.

Overall, I enjoy Polars more. There is a lot of stuff happening which is hidden behind Python syntax, which I’m not a big fan of. Polars is very explicit about what it does, which I like more.