我需要将图中的条形对齐到 matplotlib 中 x 轴刻度线之间的中心。我尝试使用align='edge'将条形移向刻度线边缘的选项,但不会在刻度线之间对齐它,我也不愿意加宽条形,以便条形之间没有空白。
我有以下形式的数据:
import matplotlib.pyplot as plt
import numpy as np
data = np.asarray([['7 Jan.', 60000],
['14 Jan.', 37000],
['21 Jan.', 32000]])
Run Code Online (Sandbox Code Playgroud)
我想用它制作一个条形图:
x = data[:, 0]
y = data[:, 1].astype(np.int)
plt.bar(x, y, width=0.7, color='#A90000')
plt.show()
Run Code Online (Sandbox Code Playgroud)
其中产生:
此外,还有this tutorial似乎解决了这个问题,但我无法使用我的代码。任何帮助表示赞赏。
您正在绘制分类数据。那是一些像["Apple", "Banana", "Cherry"]. 现在,如果您想在类别之间设置条形,则不太清楚那将是什么单位。"Apple"和之间的中间是什么"Banana"?也许"Aubergine"?但谁也不能知道。
所以最好放弃分类图。现在您当然可以将类别映射到数字。所以第一类对应于0,第二类对应于1等等。然后很容易将条形放在中间的中间,在0.5, 1.5, ...。
import matplotlib.pyplot as plt
import numpy as np
data = np.asarray([['7 Jan.', 60000],
['14 Jan.', 37000],
['21 Jan.', 32000]])
cats = data[:, 0]
x = np.arange(len(cats)) + 0.5
y = data[:, 1].astype(np.int)
plt.bar(x, y, width=0.7, color='#A90000')
plt.xticks(x-0.5, cats)
plt.show()
Run Code Online (Sandbox Code Playgroud)
不知何故,情节看起来好像缺少最后一个标签,但它根本没有出现在您的列表中。如果需要,您当然可以手动添加它。
另一种方法是使用实际日期。由于您的类别实际上对应于应该可能的日期。
from datetime import datetime
import matplotlib.pyplot as plt
import numpy as np
data = np.asarray([['7 Jan.', 60000],
['14 Jan.', 37000],
['21 Jan.', 32000]])
# convert categories to dates
xt = [datetime.strptime(d, "%d %b.") for d in data[:, 0]]
# assume all dates are equally spaced
dt = (xt[1] - xt[0])/2
# get bar positions by adding half the interval to each date
x = [tp + dt for tp in xt]
y = data[:, 1].astype(np.int)
plt.bar(x, y, width=(0.7*2*dt).days, color='#A90000')
plt.xticks(xt, data[:, 0])
plt.show()
Run Code Online (Sandbox Code Playgroud)
结果图在视觉上与上面完全相同,但是,xaxis 现在以日期时间为单位。这允许使用代码和格式器而不是手动放置刻度和标签。
import numpy as np
from datetime import datetime
import matplotlib.pyplot as plt
import matplotlib.dates as mdates
data = np.asarray([['7 Jan.', 60000],
['14 Jan.', 37000],
['21 Jan.', 32000]])
# convert categories to dates
xt = [datetime.strptime(d, "%d %b.") for d in data[:, 0]]
# assume all dates are equally spaced
dt = (xt[1] - xt[0])/2
# get bar positions by adding half the interval to each date
x = [tp + dt for tp in xt]
y = data[:, 1].astype(np.int)
plt.bar(x, y, width=(0.7*2*dt).days, color='#A90000')
plt.gca().xaxis.set_major_locator(mdates.WeekdayLocator(mdates.SU, interval=1))
plt.gca().xaxis.set_major_formatter(mdates.DateFormatter("%d %b %Y"))
plt.show()
Run Code Online (Sandbox Code Playgroud)
现在标签对应于可以以所需方式格式化的实际日期。您会注意到这里的日期是从 1900 年开始的,因此我们需要在每个星期日显示标签(因为 1900 年 1 月 7 日是星期日)。您可能希望将真实年份添加到您的数据中,以便该图变得正确。