Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Create a 100 % stacked area chart with matplotlib

I was wondering how to create a 100 % stacked area chart in matplotlib. At the matplotlib page I couldn't find an example for it.

Somebody here can show me how to achieve that?

like image 226
Thomas Becker Avatar asked Jun 01 '13 17:06

Thomas Becker


People also ask

What is a 100% stacked area chart?

A 100% Stacked Area Chart is a built-in Excel chart type, with data plotted as areas and stacked so that the cumulative area always represents 100%.

How do I create a stacked area chart?

Creating a Stacked Area Chart Select the entire dataset (A1:D6) Click the Insert tab. In the Chart group, click on the 'Insert Line or Area Chart' icon. In the 2-D Area category, click on Stacked Area.


1 Answers

A simple way to achieve this is to make sure that for every x-value, the y-values sum to 100.

I assume that you have the y-values organized in an array as in the example below, i.e.

y = np.array([[17, 19,  5, 16, 22, 20,  9, 31, 39,  8],
              [46, 18, 37, 27, 29,  6,  5, 23, 22,  5],
              [15, 46, 33, 36, 11, 13, 39, 17, 49, 17]])

To make sure the column totals are 100, you have to divide the y array by its column sums, and then multiply by 100. This makes the y-values span from 0 to 100, making the "unit" of the y-axis percent. If you instead want the values of the y-axis to span the interval from 0 to 1, don't multiply by 100.

Even if you don't have the y-values organized in one array as above, the principle is the same; the corresponding elements in each array consisting of y-values (e.g. y1, y2 etc.) should sum to 100 (or 1).

The below code is a modified version of the example @LogicalKnight linked to in his comment.

import numpy as np
from matplotlib import pyplot as plt

fnx = lambda : np.random.randint(5, 50, 10)
y = np.row_stack((fnx(), fnx(), fnx()))
x = np.arange(10)

# Make new array consisting of fractions of column-totals,
# using .astype(float) to avoid integer division
percent = y /  y.sum(axis=0).astype(float) * 100 

fig = plt.figure()
ax = fig.add_subplot(111)

ax.stackplot(x, percent)
ax.set_title('100 % stacked area chart')
ax.set_ylabel('Percent (%)')
ax.margins(0, 0) # Set margins to avoid "whitespace"

plt.show()

This gives the output shown below.

enter image description here

like image 114
sodd Avatar answered Oct 02 '22 01:10

sodd