mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-19 22:51:41 +08:00
adaa79e74c
sources/tech/20200610 Concise data plotting in Python with Altair.md
194 lines
7.9 KiB
Markdown
194 lines
7.9 KiB
Markdown
[#]: collector: (lujun9972)
|
||
[#]: translator: ( )
|
||
[#]: reviewer: ( )
|
||
[#]: publisher: ( )
|
||
[#]: url: ( )
|
||
[#]: subject: (Concise data plotting in Python with Altair)
|
||
[#]: via: (https://opensource.com/article/20/6/altair-python)
|
||
[#]: author: (Shaun Taylor-Morgan https://opensource.com/users/shaun-taylor-morgan)
|
||
|
||
Concise data plotting in Python with Altair
|
||
======
|
||
Altair offers an elegant interface and its own plotting language in a
|
||
Python libraries
|
||
![metrics and data shown on a computer screen][1]
|
||
|
||
The [plotting libraries][2] available in Python offer multiple ways to present data according to your preferences, whether you prize flexibility, design, ease-of-use, or a particular style.
|
||
|
||
Plotting in Altair delivers a different solution than others which I find to be simpler as the amount of plotting done scales. It has an elegant interface, thanks to [Vega][3], the declarative plotting language that allows you to define what your plots should look like, rather than writing loops and conditionals to construct them.
|
||
|
||
### Fluent in plotting
|
||
|
||
I compare each Python plotting library by plotting the same multi-bar plot. Before we go further, note that you may need to tune your Python environment to get this code to run, including the following.
|
||
|
||
* Installing a recent version of Python (instructions for [Linux][4], [Mac][5], and [Windows][6])
|
||
* Verify that version of Python works with the libraries used in the tutorial
|
||
|
||
|
||
|
||
The data is available online and can be imported using pandas:
|
||
|
||
|
||
```
|
||
import pandas as pd
|
||
df = pd.read_csv('<https://anvil.works/blog/img/plotting-in-python/uk-election-results.csv>')
|
||
```
|
||
|
||
Now we're ready to go. As a point of comparison, this is the plot we made in [Matplotlib][7]:
|
||
|
||
![Matplotlib UK election results][8]
|
||
|
||
The Matplotlib plot took 16 lines of code to create, including manually calculating the positions of each bar.
|
||
|
||
Here's how to make a similar plot in Altair:
|
||
|
||
|
||
```
|
||
import altair as alt
|
||
|
||
chart = alt.Chart(df).mark_bar().encode(
|
||
x='party',
|
||
y='seats',
|
||
column='year',
|
||
color='party',
|
||
)
|
||
|
||
chart.save('altair-elections.html')
|
||
```
|
||
|
||
Much more concise! Just like [Seaborn][9], Altair works with data that has one column per variable ([Long Form][10]). This allows you to map each variable onto an aspect of the plot—Altair calls these aspects "channels." In our case, we want one bar per `party` on the `x`-axis, we want the `seats` each party won on the `y`-axis, and we want to group the bars into `column`s by `year`. We also want to `color` the bars by `party`. That's how you would describe it in words, and it's exactly what the code says!
|
||
|
||
Here's what the plot looks like:
|
||
|
||
![Altair plot with default styling][11]
|
||
|
||
### Tweaking the style
|
||
|
||
That's not too far from what we want. The main difference from the Matplotlib plot is that each Altair `year` group is displayed with a little white space in between—this is just a feature of Altair's multi-bar plots, and it's not a problem.
|
||
|
||
However, there are a few other little style improvements that we do want to make.
|
||
|
||
#### Non-integer values
|
||
|
||
The two non-integer year names (`Feb 1974` and `Oct 1974`) are displayed as `NaN`. We can fix these by casting our `year` values to strings:
|
||
|
||
|
||
```
|
||
` df['year'] = df['year'].astype(str)`
|
||
```
|
||
|
||
#### Specifying sort order
|
||
|
||
We then also need to tell Altair how to sort the data. Altair allows us to specify more details about the `column` channel by passing it a `Column` object. So we tell it to sort in the order that the data appears in the DataFrame:
|
||
|
||
|
||
```
|
||
chart = alt.Chart(df).mark_bar().encode(
|
||
# ...
|
||
column=alt.Column('year', sort=list(df['year']), title=None),
|
||
# ...
|
||
)
|
||
```
|
||
|
||
#### Removing axis titles
|
||
|
||
We have removed the "year" label from the top of the plot by setting `title=None`. Let's also remove the "party" labels from each column:
|
||
|
||
|
||
```
|
||
chart = alt.Chart(df).mark_bar().encode(
|
||
x=alt.X('party', title=None),
|
||
# ...
|
||
)
|
||
```
|
||
|
||
#### Specifying a colormap
|
||
|
||
Finally, we want to specify our own colors to use for the bars. Altair lets us specify a mapping between values in a `domain` and colors in a `range`, which is exactly what we need:
|
||
|
||
|
||
```
|
||
cmap = {
|
||
'Conservative': '#0343df',
|
||
'Labour': '#e50000',
|
||
'Liberal': '#ffff14',
|
||
'Others': '#929591',
|
||
}
|
||
|
||
chart = alt.Chart(df).mark_bar().encode(
|
||
# ...
|
||
color=alt.Color('party', scale=alt.Scale(domain=list(cmap.keys()), range=list(cmap.values())))
|
||
)
|
||
```
|
||
|
||
#### Final code with style tweaks
|
||
|
||
After applying these styling tweaks, our code is a little less pleasing to the eye, but it still uses the declarative approach that makes Altair so scalable. We're still assigning independent variables from our data to separate aspects of our plot, rather than performing complex data manipulations as we often need to do in Matplotlib. The only difference is that our variable names are now wrapped in objects such as `alt.X()` that let us tweak how they appear:
|
||
|
||
|
||
```
|
||
import altair as alt
|
||
from votes import long as df
|
||
|
||
cmap = {
|
||
'Conservative': '#0343df',
|
||
'Labour': '#e50000',
|
||
'Liberal': '#ffff14',
|
||
'Others': '#929591',
|
||
}
|
||
|
||
df['year'] = df['year'].astype(str)
|
||
|
||
# We're still assigning, e.g. 'party' to x, but now we've wrapped it
|
||
# in alt.X in order to specify its styling
|
||
chart = alt.Chart(df).mark_bar().encode(
|
||
x=alt.X('party', title=None),
|
||
y='seats',
|
||
column=alt.Column('year', sort=list(df['year']), title=None),
|
||
color=alt.Color('party', scale=alt.Scale(domain=list(cmap.keys()), range=list(cmap.values())))
|
||
)
|
||
|
||
chart.save('altair-elections.html')
|
||
```
|
||
|
||
In fairness to Matplotlib, we've now reached the same number of lines of code (16) as we used there!
|
||
|
||
Here's the Altair plot with our styling tweaks applied:
|
||
|
||
![The Altair plot with our custom styling][12]
|
||
|
||
### Conclusion** **
|
||
|
||
While the amount of code it takes to plot using Altair is similar to other libraries, its declarative plotting language adds a layer of sophistication I have grown to appreciate. Altair also offers a cleanly separate way to tune and tweak the style, which keeps that syntax out of the code blocks meant for plotting. Altair is another great library when considering a plotting solution for Python.
|
||
|
||
\---
|
||
|
||
_This article was first shared [here][13] and is edited and republished with permission._
|
||
|
||
--------------------------------------------------------------------------------
|
||
|
||
via: https://opensource.com/article/20/6/altair-python
|
||
|
||
作者:[Shaun Taylor-Morgan][a]
|
||
选题:[lujun9972][b]
|
||
译者:[译者ID](https://github.com/译者ID)
|
||
校对:[校对者ID](https://github.com/校对者ID)
|
||
|
||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||
|
||
[a]: https://opensource.com/users/shaun-taylor-morgan
|
||
[b]: https://github.com/lujun9972
|
||
[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/metrics_data_dashboard_system_computer_analytics.png?itok=oxAeIEI- (metrics and data shown on a computer screen)
|
||
[2]: https://opensource.com/article/20/4/plot-data-python
|
||
[3]: https://vega.github.io/vega/
|
||
[4]: https://opensource.com/article/20/4/install-python-linux
|
||
[5]: https://opensource.com/article/19/5/python-3-default-mac
|
||
[6]: https://opensource.com/article/19/8/how-install-python-windows
|
||
[7]: https://opensource.com/article/20/5/matplotlib-python
|
||
[8]: https://opensource.com/sites/default/files/uploads/matplotlib_1_1.png (Matplotlib UK election results)
|
||
[9]: https://anvil.works/blog/plotting-in-seaborn
|
||
[10]: https://anvil.works/blog/tidy-data
|
||
[11]: https://opensource.com/sites/default/files/uploads/altair-first-try.png (Altair plot with default styling)
|
||
[12]: https://opensource.com/sites/default/files/uploads/altair_3.png (The Altair plot with our custom styling.)
|
||
[13]: https://anvil.works/blog/plotting-in-altair
|