【Python】模块学习之matplotlib绘制柱状图、饼状图、动态图及解决中文显示问题

前言

原文链接:http://www.cnblogs.com/Detector/p/8850280.html

通过数据绘图,我们可以将枯燥的数字转换成容易被人们接受的图表,从而让人留下更加深刻的印象。

matplotlib是基于numpy的一套Python工具包。这个包提供了丰富的数据绘图工具,主要用于绘制一些2D统计图形。它可以以各种硬拷贝格式在跨平台的交互式环境生成出版质量级别的图形。

通过 Matplotlib,可以仅需要几行代码,便可以生成绘图,直方图,功率谱,条形图,错误图,散点图,甚至动态图等。

下面是他们官网的一些例子:

https://matplotlib.org/gallery

博主也是在测试一个算法的时候,想更直观的呈现数据走向,学习其相关的用法,此篇为学习记录。

安装

Matplotlib得安装比较容易,可以直接通过pip安装,也可以通过下载安装包的方式安装。

pip3.6 install matplotlib

也可以直接下载安装包进行安装,如果你的Python是其他版本,则点击其他matplotlib下载对应Python版本
matplotlib-1.4.3.win-amd64-py3.4.exe 其他matplotlib

柱状图

用python画柱状图很容易,主要的方法是:

atplotlib.pyplot.bar(left, height, width=0.8, bottom=None, hold=None, data=None, **kwargs)

参数说明:

left: 每一个柱形左侧的X坐标

height:每一个柱形的高度

width: 柱形之间的宽度

bottom: 柱形的Y坐标

color: 柱形的颜色

示例代码:

import matplotlib.pyplot as plt
import time

source_data = {'mock_verify': 369, 'mock_notify': 192, 'mock_sale': 517}  # 设置原始数据

for a, b in source_data.items():
    plt.text(a, b + 0.05, '%.0f' % b, ha='center', va='bottom', fontsize=11)  # ha 文字指定在柱体中间, va指定文字位置 fontsize指定文字体大小

# 设置X轴Y轴数据,两者都可以是list或者tuple
x_axis = tuple(source_data.keys())
y_axis = tuple(source_data.values())
plt.bar(x_axis, y_axis, color='rgb')  # 如果不指定color,所有的柱体都会是一个颜色

plt.xlabel(u"渠道名")  # 指定x轴描述信息
plt.ylabel(u"访问量")  # 指定y轴描述信息
plt.title("渠道访问量统计表")  # 指定图表描述信息
plt.ylim(0, 600)  # 指定Y轴的高度
plt.savefig('{}.png'.format(time.strftime('%Y%m%d%H%M%S')))  # 保存为图片
plt.show()

效果图:

解决中文显示问题

从上面的图片看到,matplotlib绘制图像显示中文时,中文会变成小方格子。搜了很多资料,最佳的解决方法是动态设置参数,添加下面的代码。

from pylab import mpl

mpl.rcParams['font.sans-serif'] = ['FangSong'] # 指定默认字体
mpl.rcParams['axes.unicode_minus'] = False # 解决保存图像是负号'-'显示为方块的问题

修改后的图片

饼状图

用python画柱状图很容易,主要的方法是:

plt.pie(x=fracs, labels=labels, explode=explode, autopct='%3.1f %%',shadow=False, labeldistance=1.1, startangle=0, pctdistance=0.8)

示例代码:

import matplotlib.pyplot as plt
import time
source_data = {'var': 369, 'plt': 192, 'sale': 115, 'a': 50, 'b': 45, 'c':      40, 'd': 35, 'e': 30, 'f': 25, 'g': 20,'h': 18, 'i': 16, 'j': 14, 'k':      10, 'l': 8, 'm': 7, 'n': 6, 'o': 6, 'p': 6, 'q': 6, 'r': 6, 's': 6,         't':      6, 'u': 6, 'v': 6,'w': 6, 'x': 6, 'y': 6, 'z': 6}  # 设置原始数据
labels = list(source_data.keys())  # 设置标签
fracs = [x for x in source_data.values()]
explode = [x*0.03 for x in range(len(source_data))]  # 与labels一一对应,数值越大离中心区越远
plt.axes(aspect=1)  # 设置X轴 Y轴比例

# labeldistance标签离中心距离  pctdistance百分百数据离中心区距离 autopct 百分比的格式 shadow阴影
plt.pie(x=fracs, labels=labels, explode=explode, autopct='%3.1f %%',
        shadow=False, labeldistance=1.1, startangle=0, pctdistance=0.8, center=(-1, 0))  

# 控制位置:bbox_to_anchor数组中,前者控制左右移动,后者控制上下。ncol控制 图例所列的列数。默认值为1。
plt.legend(loc=7, bbox_to_anchor=(1.2, 0.80), ncol=3, fancybox=True, shadow=True, fontsize=8)  
plt.show()

效果图:

实时呈现动态数据

用python画动态图主要的方法是:

ani = FuncAnimation(fig, update, frames=np.linspace(0, 2*np.pi, 128),
                    init_func=init, blit=True)

官方使用介绍:
https://matplotlib.org/api/animation_api.html#id4

示例代码:

import numpy as np
import matplotlib.pyplot as plt
import matplotlib.animation as animation
import time
# Fixing random state for reproducibility
np.random.seed(196)
# 初始数据绘图
dis = np.zeros(40)
dis2 = dis
fig, ax = plt.subplots()
line, = ax.plot(dis)
ax.set_ylim(-1, 1)
plt.grid(True)
ax.set_ylabel("distance: m")
ax.set_xlabel("time")

def update(frame):
    global dis
    global dis2
    global line
    # 读入模拟
    a = np.random.rand() * 2 - 1
    time.sleep(np.random.rand() / 10)
    # 绘图数据生成
    dis[0:-1] = dis2[1:]
    dis[-1] = a
    dis2 = dis
    # 绘图
    line.set_ydata(dis)
    # 颜色设置
    plt.setp(line, 'color', 'c', 'linewidth', 2.0)
ani = animation.FuncAnimation(fig, update, frames=None, interval=100)
plt.show()

效果图:

附录

matplotlib中对显示的字体和颜色都可以进行定制:

其他颜色可以参考下面的链接:
matplotlib的颜色及线条控制

相关资料:
https://blog.csdn.net/u010758410/article/details/71743225

https://blog.csdn.net/jenyzhang/article/details/52047557

https://www.cnblogs.com/darkknightzh/p/6117528.html

https://www.zhihu.com/question/49481680/answer/344653927

猜你喜欢

转载自blog.csdn.net/Detector_/article/details/80056690