我正在尝试阻止注释文本在我的图表中重叠。 Matplotlib overlapping annotations 的已接受答案中建议的方法看起来非常有前途,但适用于条形图。我无法将“轴”方法转换为我想要做的事情,而且我不明白文本是如何排列的。
import sys
import matplotlib.pyplot as plt
# start new plot
plt.clf()
plt.xlabel("Proportional Euclidean Distance")
plt.ylabel("Percentage Timewindows Attended")
plt.title("Test plot")
together = [(0, 1.0, 0.4), (25, 1.0127692669427917, 0.41), (50, 1.016404709797609, 0.41), (75, 1.1043426359673716, 0.42), (100, 1.1610446924342996, 0.44), (125, 1.1685687930691457, 0.43), (150, 1.3486407784550272, 0.45), (250, 1.4013999168008104, 0.45)]
together.sort()
for x,y,z in together:
plt.annotate(str(x), xy=(y, z), size=8)
eucs = [y for (x,y,z) in together]
covers = [z for (x,y,z) in together]
p1 = plt.plot(eucs,covers,color="black", alpha=0.5)
plt.savefig("test.png")
图片(如果可行)可以找到here (这段代码):
和here (更复杂):
最佳答案
我只是想在这里发布另一个解决方案,我写的一个小库来实现这种事情:https://github.com/Phlya/adjustText 可以在此处看到该过程的示例:
这是示例图片:
import matplotlib.pyplot as plt
from adjustText import adjust_text
import numpy as np
together = [(0, 1.0, 0.4), (25, 1.0127692669427917, 0.41), (50, 1.016404709797609, 0.41), (75, 1.1043426359673716, 0.42), (100, 1.1610446924342996, 0.44), (125, 1.1685687930691457, 0.43), (150, 1.3486407784550272, 0.45), (250, 1.4013999168008104, 0.45)]
together.sort()
text = [x for (x,y,z) in together]
eucs = [y for (x,y,z) in together]
covers = [z for (x,y,z) in together]
p1 = plt.plot(eucs,covers,color="black", alpha=0.5)
texts = []
for x, y, s in zip(eucs, covers, text):
texts.append(plt.text(x, y, s))
plt.xlabel("Proportional Euclidean Distance")
plt.ylabel("Percentage Timewindows Attended")
plt.title("Test plot")
adjust_text(texts, only_move={'points':'y', 'texts':'y'}, arrowprops=dict(arrowstyle="->", color='r', lw=0.5))
plt.show()
如果你想要一个完美的身材,你可以稍微摆弄一下。首先,让我们也让文本排斥线条 - 为此我们只需使用 scipy.interpolate.interp1d 沿着它们创建许多虚拟点。
我们希望避免沿 x 轴移动标签,因为,为什么不这样做是为了便于说明。为此,我们使用参数 only_move={'points':'y', 'text':'y'}
。如果我们只想在它们与文本重叠的情况下沿 x 轴移动它们,请使用 move_only={'points':'y', 'text':'xy'}
。同样在开始时,该函数选择文本相对于其原始点的最佳对齐方式,因此我们也只希望沿 y 轴发生这种情况,因此 autoalign='y'
。我们还减少了点的排斥力,以避免由于我们人为地避免线条而导致文本飞得太远。一起来:
from scipy import interpolate
p1 = plt.plot(eucs,covers,color="black", alpha=0.5)
texts = []
for x, y, s in zip(eucs, covers, text):
texts.append(plt.text(x, y, s))
f = interpolate.interp1d(eucs, covers)
x = np.arange(min(eucs), max(eucs), 0.0005)
y = f(x)
plt.xlabel("Proportional Euclidean Distance")
plt.ylabel("Percentage Timewindows Attended")
plt.title("Test plot")
adjust_text(texts, x=x, y=y, autoalign='y',
only_move={'points':'y', 'text':'y'}, force_points=0.15,
arrowprops=dict(arrowstyle="->", color='r', lw=0.5))
plt.show()
关于python - 如何修复重叠的注释/文本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19073683/