Matplotlib 高级绘图函数
除 plot() 和 bar() 等基本绘图函数外,Matplotlib 还提供了多种专用绘图函数,用于阶梯图、茎叶图、堆叠面积图、极坐标图等场景。
函数一览
| 函数 | 功能 |
|---|---|
| step() | 绘制阶梯图(数据分段常数变化) |
| stem() | 绘制茎叶图(垂直线+顶端标记) |
| eventplot() | 绘制事件图(多条水平线标记事件位置) |
| stackplot() | 绘制堆叠面积图 |
| broken_barh() | 绘制水平断条图(甘特图风格) |
| vlines() / hlines() | 绘制多条垂直/水平参考线 |
| fill() | 绘制填充多边形 |
| polar() | 在极坐标系中绘图 |
| loglog() / semilogx() / semilogy() | 对数坐标折线图 |
| grouped_bar() | 分组柱状图(3.10+ 新功能) |
| bar_label() / pie_label() | 给柱状图/饼图添加数值标签 |
step() - 阶梯图
matplotlib.pyplot.step(x, y, *args, where='pre', **kwargs)
| 参数 | 说明 |
|---|---|
| x, y | 数据点坐标 |
| where | 阶梯位置:'pre'(前)、'post'(后)、'mid'(中) |
实例
import matplotlib.pyplot as plt
import numpy as np
x = np.arange(0, 10, 0.5)
y = np.sin(x)
fig, ax = plt.subplots(figsize=(8, 4), layout='constrained')
# 对比三种 where 模式
ax.plot(x, y, 'gray', alpha=0.3, label='Original (plot)')
ax.step(x, y, where='pre', label='where="pre"', linewidth=2)
ax.step(x, y, where='post', label='where="post"', linewidth=2)
ax.step(x, y, where='mid', label='where="mid"', linewidth=2)
ax.set_title('step() - Staircase Plot')
ax.legend()
ax.grid(True, alpha=0.3)
plt.show()
import numpy as np
x = np.arange(0, 10, 0.5)
y = np.sin(x)
fig, ax = plt.subplots(figsize=(8, 4), layout='constrained')
# 对比三种 where 模式
ax.plot(x, y, 'gray', alpha=0.3, label='Original (plot)')
ax.step(x, y, where='pre', label='where="pre"', linewidth=2)
ax.step(x, y, where='post', label='where="post"', linewidth=2)
ax.step(x, y, where='mid', label='where="mid"', linewidth=2)
ax.set_title('step() - Staircase Plot')
ax.legend()
ax.grid(True, alpha=0.3)
plt.show()
stem() - 茎叶图
matplotlib.pyplot.stem(*args, linefmt=None, markerfmt=None,
basefmt=None, bottom=0, orientation='vertical', **kwargs)
实例
import matplotlib.pyplot as plt
import numpy as np
n = 30
x = np.arange(n)
y = np.random.randn(n) * 2
fig, (ax1, ax2) = plt.subplots(1, 2, figsize=(10, 4),
layout='constrained')
# 垂直茎叶图
ax1.stem(x, y, linefmt='steelblue', markerfmt='o',
basefmt='gray')
ax1.set_title('Vertical stem()')
# 水平茎叶图
ax2.stem(x, y, linefmt='coral', markerfmt='s',
basefmt='gray', orientation='horizontal')
ax2.set_title('Horizontal stem()')
plt.show()
import numpy as np
n = 30
x = np.arange(n)
y = np.random.randn(n) * 2
fig, (ax1, ax2) = plt.subplots(1, 2, figsize=(10, 4),
layout='constrained')
# 垂直茎叶图
ax1.stem(x, y, linefmt='steelblue', markerfmt='o',
basefmt='gray')
ax1.set_title('Vertical stem()')
# 水平茎叶图
ax2.stem(x, y, linefmt='coral', markerfmt='s',
basefmt='gray', orientation='horizontal')
ax2.set_title('Horizontal stem()')
plt.show()
stackplot() - 堆叠面积图
matplotlib.pyplot.stackplot(x, *args, labels=None, colors=None,
baseline='zero', **kwargs)
实例
import matplotlib.pyplot as plt
import numpy as np
x = np.arange(0, 10, 0.1)
y1 = np.sin(x) + 2
y2 = np.cos(x) + 2
y3 = 0.5 * np.sin(2*x) + 2
fig, ax = plt.subplots(figsize=(8, 5), layout='constrained')
ax.stackplot(x, y1, y2, y3,
labels=['Signal 1', 'Signal 2', 'Signal 3'],
colors=['#3498db', '#e74c3c', '#2ecc71'],
alpha=0.7)
ax.set_title('stackplot() - Stacked Area Chart')
ax.legend(loc='upper right')
plt.show()
import numpy as np
x = np.arange(0, 10, 0.1)
y1 = np.sin(x) + 2
y2 = np.cos(x) + 2
y3 = 0.5 * np.sin(2*x) + 2
fig, ax = plt.subplots(figsize=(8, 5), layout='constrained')
ax.stackplot(x, y1, y2, y3,
labels=['Signal 1', 'Signal 2', 'Signal 3'],
colors=['#3498db', '#e74c3c', '#2ecc71'],
alpha=0.7)
ax.set_title('stackplot() - Stacked Area Chart')
ax.legend(loc='upper right')
plt.show()
vlines() / hlines() - 参考线
matplotlib.pyplot.vlines(x, ymin, ymax, colors=None, **kwargs) matplotlib.pyplot.hlines(y, xmin, xmax, colors=None, **kwargs)
实例
import matplotlib.pyplot as plt
import numpy as np
x = np.linspace(0, 10, 100)
y = np.sin(x)
fig, ax = plt.subplots(figsize=(8, 4), layout='constrained')
ax.plot(x, y, 'k-', linewidth=2, alpha=0.7)
# 多条垂直线(标注 π 的倍数)
pi_x = [np.pi, 2*np.pi, 3*np.pi]
ax.vlines(pi_x, ymin=-1.5, ymax=1.5,
colors='red', linestyles='--', alpha=0.5,
label='π multiples')
# 多条水平线(标注 ±1)
ax.hlines([-1, 0, 1], xmin=0, xmax=10,
colors=['gray','gray','gray'],
linestyles=['--',':','--'], alpha=0.5)
ax.set_title('vlines() and hlines()')
ax.legend()
plt.show()
import numpy as np
x = np.linspace(0, 10, 100)
y = np.sin(x)
fig, ax = plt.subplots(figsize=(8, 4), layout='constrained')
ax.plot(x, y, 'k-', linewidth=2, alpha=0.7)
# 多条垂直线(标注 π 的倍数)
pi_x = [np.pi, 2*np.pi, 3*np.pi]
ax.vlines(pi_x, ymin=-1.5, ymax=1.5,
colors='red', linestyles='--', alpha=0.5,
label='π multiples')
# 多条水平线(标注 ±1)
ax.hlines([-1, 0, 1], xmin=0, xmax=10,
colors=['gray','gray','gray'],
linestyles=['--',':','--'], alpha=0.5)
ax.set_title('vlines() and hlines()')
ax.legend()
plt.show()
loglog() / semilogx() / semilogy() - 对数坐标
matplotlib.pyplot.loglog(*args, **kwargs) matplotlib.pyplot.semilogx(*args, **kwargs) matplotlib.pyplot.semilogy(*args, **kwargs)
实例
import matplotlib.pyplot as plt
import numpy as np
x = np.logspace(0, 3, 50) # 10^0 到 10^3
y1 = x**2
y2 = x**1.5
y3 = np.exp(x/50)
fig, axes = plt.subplots(1, 3, figsize=(12, 4),
layout='constrained')
# 双对数
axes[0].loglog(x, y1, 'b-', label='x²')
axes[0].loglog(x, y2, 'r--', label='x¹·⁵')
axes[0].set_title('loglog()')
axes[0].grid(True, alpha=0.3)
axes[0].legend()
# x 轴对数
axes[1].semilogx(x, y3, 'green', label='exp(x/50)')
axes[1].set_title('semilogx()')
axes[1].grid(True, alpha=0.3)
# y 轴对数
axes[2].semilogy(x, y3, 'purple', label='exp(x/50)')
axes[2].set_title('semilogy()')
axes[2].grid(True, alpha=0.3)
plt.show()
import numpy as np
x = np.logspace(0, 3, 50) # 10^0 到 10^3
y1 = x**2
y2 = x**1.5
y3 = np.exp(x/50)
fig, axes = plt.subplots(1, 3, figsize=(12, 4),
layout='constrained')
# 双对数
axes[0].loglog(x, y1, 'b-', label='x²')
axes[0].loglog(x, y2, 'r--', label='x¹·⁵')
axes[0].set_title('loglog()')
axes[0].grid(True, alpha=0.3)
axes[0].legend()
# x 轴对数
axes[1].semilogx(x, y3, 'green', label='exp(x/50)')
axes[1].set_title('semilogx()')
axes[1].grid(True, alpha=0.3)
# y 轴对数
axes[2].semilogy(x, y3, 'purple', label='exp(x/50)')
axes[2].set_title('semilogy()')
axes[2].grid(True, alpha=0.3)
plt.show()
eventplot() - 事件图
matplotlib.pyplot.eventplot(positions, orientation='horizontal',
lineoffsets=1, linelengths=1, linewidths=None, colors=None, **kwargs)
实例
import matplotlib.pyplot as plt
import numpy as np
np.random.seed(42)
# 三个通道的事件时间
events_A = np.random.rand(20) * 10
events_B = np.random.rand(15) * 10
events_C = np.random.rand(25) * 10
fig, ax = plt.subplots(figsize=(8, 4), layout='constrained')
ax.eventplot([events_A, events_B, events_C],
colors=['#e74c3c', '#3498db', '#2ecc71'],
lineoffsets=[0, 1, 2], # 每条线的垂直位置
linelengths=0.8,
linewidths=1.5)
ax.set_title('eventplot() - Event Timing Diagram')
ax.set_xlabel('Time (s)')
ax.set_yticks([0, 1, 2])
ax.set_yticklabels(['Channel A', 'Channel B', 'Channel C'])
plt.show()
import numpy as np
np.random.seed(42)
# 三个通道的事件时间
events_A = np.random.rand(20) * 10
events_B = np.random.rand(15) * 10
events_C = np.random.rand(25) * 10
fig, ax = plt.subplots(figsize=(8, 4), layout='constrained')
ax.eventplot([events_A, events_B, events_C],
colors=['#e74c3c', '#3498db', '#2ecc71'],
lineoffsets=[0, 1, 2], # 每条线的垂直位置
linelengths=0.8,
linewidths=1.5)
ax.set_title('eventplot() - Event Timing Diagram')
ax.set_xlabel('Time (s)')
ax.set_yticks([0, 1, 2])
ax.set_yticklabels(['Channel A', 'Channel B', 'Channel C'])
plt.show()
broken_barh() - 断条图
matplotlib.pyplot.broken_barh(xranges, yrange, **kwargs)
实例
import matplotlib.pyplot as plt
# 模拟甘特图
tasks = [
[(0, 5)], # Task 1: 从0开始,持续5
[(3, 4)], # Task 2: 从3开始,持续4
[(2, 3), (6, 3)], # Task 3: 两个时间段
[(5, 4)], # Task 4
[(1, 2), (4, 2), (8, 2)], # Task 5: 三个时间段
]
fig, ax = plt.subplots(figsize=(8, 4), layout='constrained')
colors = ['#3498db', '#e74c3c', '#2ecc71', '#f39c12', '#9b59b6']
for i, (task, color) in enumerate(zip(tasks, colors)):
ax.broken_barh(task, (i - 0.4, 0.8), facecolors=color)
ax.set_title('broken_barh() - Gantt-style Chart')
ax.set_xlabel('Time')
ax.set_yticks(range(5))
ax.set_yticklabels([f'Task {i+1}' for i in range(5)])
ax.grid(True, alpha=0.3)
plt.show()
# 模拟甘特图
tasks = [
[(0, 5)], # Task 1: 从0开始,持续5
[(3, 4)], # Task 2: 从3开始,持续4
[(2, 3), (6, 3)], # Task 3: 两个时间段
[(5, 4)], # Task 4
[(1, 2), (4, 2), (8, 2)], # Task 5: 三个时间段
]
fig, ax = plt.subplots(figsize=(8, 4), layout='constrained')
colors = ['#3498db', '#e74c3c', '#2ecc71', '#f39c12', '#9b59b6']
for i, (task, color) in enumerate(zip(tasks, colors)):
ax.broken_barh(task, (i - 0.4, 0.8), facecolors=color)
ax.set_title('broken_barh() - Gantt-style Chart')
ax.set_xlabel('Time')
ax.set_yticks(range(5))
ax.set_yticklabels([f'Task {i+1}' for i in range(5)])
ax.grid(True, alpha=0.3)
plt.show()
fill() - 填充多边形
matplotlib.pyplot.fill(*args, **kwargs)
实例
import matplotlib.pyplot as plt
import numpy as np
fig, ax = plt.subplots(layout='constrained')
# 填充一个菱形
x = [1, 2, 1, 0]
y = [0, 1, 2, 1]
ax.fill(x, y, alpha=0.5, color='steelblue',
edgecolor='black', label='Diamond')
# 填充一个五边形
theta = np.linspace(0, 2*np.pi, 6)
x2 = 2.5 + np.cos(theta)
y2 = 1 + np.sin(theta)
ax.fill(x2, y2, alpha=0.5, color='coral',
edgecolor='black', label='Pentagon')
ax.set_title('fill() - Filled Polygons')
ax.set_aspect('equal')
ax.legend()
ax.grid(True, alpha=0.3)
plt.show()
import numpy as np
fig, ax = plt.subplots(layout='constrained')
# 填充一个菱形
x = [1, 2, 1, 0]
y = [0, 1, 2, 1]
ax.fill(x, y, alpha=0.5, color='steelblue',
edgecolor='black', label='Diamond')
# 填充一个五边形
theta = np.linspace(0, 2*np.pi, 6)
x2 = 2.5 + np.cos(theta)
y2 = 1 + np.sin(theta)
ax.fill(x2, y2, alpha=0.5, color='coral',
edgecolor='black', label='Pentagon')
ax.set_title('fill() - Filled Polygons')
ax.set_aspect('equal')
ax.legend()
ax.grid(True, alpha=0.3)
plt.show()
Matplotlib 参考文档
点我分享笔记