ChatGPT解决这个技术问题 Extra ChatGPT

在 matplotlib 中更改 x 或 y 轴上的“滴答频率”

我正在尝试修复 python 如何绘制我的数据。说:

x = [0,5,9,10,15]
y = [0,1,2,3,4]

matplotlib.pyplot.plot(x,y)
matplotlib.pyplot.show()

x 轴的刻度以 5 的间隔绘制。有没有办法让它显示 1 的间隔?

虽然刻度在这里是合适的词,但将刻度更改为步长肯定会引导更多新手回答这个问题。
密切相关的问题:stackoverflow.com/questions/6682784/… 和一个很好的解决方案:pyplot.locator_params(nbins=4)
我已经回滚了编辑,因为没有必要彻底改变规范问题。这个问题在以前的形式中已经被找到了 130 万次。
@tdy 谢谢,我几乎做了同样的事情。 (我们需要的最后一件事是断言不需要示例)

u
unutbu

您可以使用 plt.xticks 显式设置要打勾的位置:

plt.xticks(np.arange(min(x), max(x)+1, 1.0))

例如,

import numpy as np
import matplotlib.pyplot as plt

x = [0,5,9,10,15]
y = [0,1,2,3,4]
plt.plot(x,y)
plt.xticks(np.arange(min(x), max(x)+1, 1.0))
plt.show()

(使用 np.arange 而不是 Python 的 range 函数,以防 min(x)max(x) 是浮点数而不是整数。)

plt.plot(或 ax.plot)函数将自动设置默认的 xy 限制。如果您希望保持这些限制,并且只更改刻度线的步长,那么您可以使用 ax.get_xlim() 来发现 Matplotlib 已经设置的限制。

start, end = ax.get_xlim()
ax.xaxis.set_ticks(np.arange(start, end, stepsize))

默认的刻度格式化程序应该可以将刻度值四舍五入到合理的有效数字位数。但是,如果您希望对格式有更多的控制,您可以定义自己的格式化程序。例如,

ax.xaxis.set_major_formatter(ticker.FormatStrFormatter('%0.1f'))

这是一个可运行的示例:

import numpy as np
import matplotlib.pyplot as plt
import matplotlib.ticker as ticker

x = [0,5,9,10,15]
y = [0,1,2,3,4]
fig, ax = plt.subplots()
ax.plot(x,y)
start, end = ax.get_xlim()
ax.xaxis.set_ticks(np.arange(start, end, 0.712123))
ax.xaxis.set_major_formatter(ticker.FormatStrFormatter('%0.1f'))
plt.show()

有没有办法让它仍然决定它自己的限制,而只是改变步长?如果 min 是 3523.232512 之类的,这种方法不是很好!
@Corone,你问已经有一段时间了,但我在下面发布了一个答案,可以在仍然使用自动边界确定的同时轻松控制步长。
请注意,plt.xticks(np.arange(min(x), max(x)+1, 1.0)) 中的 +1 是显示最后一个刻度线所必需的。
是的,np.arange(start, stop)半开 区间 [start, stop) 中生成值,包括 start 但不包括 stop。所以我使用 max(x)+1 来确保包含 max(x)
是否有日期时间的等价物,例如 plt.xticks(np.arange(min(dates), max(dates)+0.1,0.1) ?似乎只绘制年份
r
robochat

另一种方法是设置轴定位器:

import matplotlib.ticker as plticker

loc = plticker.MultipleLocator(base=1.0) # this locator puts ticks at regular intervals
ax.xaxis.set_major_locator(loc)

根据您的需要,有几种不同类型的定位器。

这是一个完整的例子:

import matplotlib.pyplot as plt
import matplotlib.ticker as plticker

x = [0,5,9,10,15]
y = [0,1,2,3,4]
fig, ax = plt.subplots()
ax.plot(x,y)
loc = plticker.MultipleLocator(base=1.0) # this locator puts ticks at regular intervals
ax.xaxis.set_major_locator(loc)
plt.show()

这不能按预期工作。具体来说,在使用日期时,它不会使用适当的日期。
使用日期时,您应该使用 matplotlib.dates 模块中的方法。例如 matplotlib.dates.AutoDateLocator()
它按我的预期工作,有日期。这种解决方案比公认的解决方案容易得多。
base=1.0 实际上是什么意思/做什么?
base=1.0 表示每个整数都有一个定位器。文档说 MultipleLocator “在视图间隔内为每个基数的整数倍设置一个刻度。”。所以如果base=2,那么偶数会有一个勾号,我认为你可以设置base=2.5。
j
jthomas

我喜欢这个解决方案(来自 Matplotlib Plotting Cookbook):

import matplotlib.pyplot as plt
import matplotlib.ticker as ticker

x = [0,5,9,10,15]
y = [0,1,2,3,4]

tick_spacing = 1

fig, ax = plt.subplots(1,1)
ax.plot(x,y)
ax.xaxis.set_major_locator(ticker.MultipleLocator(tick_spacing))
plt.show()

这个解决方案让您可以通过 ticker.MultipleLocater() 的数字明确控制刻度间距,允许自动确定限制,并且以后易于阅读。


一种无需明确计算刻度的方法!
这与 this one 的答案相同。两年后添加相同的答案是没有意义的。
接得好。当我发布答案时,我没有认识到它们是一样的。不过,我认为这个演示文稿更容易理解。
此答案中的书籍参考还提供了有关更多信息的有用来源。
这与三年前出现的机器人聊天的答案相同。
g
glopes

如果有人对通用单线感兴趣,只需获取当前刻度并使用它通过每隔一个刻度采样来设置新刻度。

ax.set_xticks(ax.get_xticks()[::2])

这是不同刻度类型(str、float、datetime)的唯一通用答案
删除非整数刻度:ax.set_xticks([tick for tick in ax.get_xticks() if tick % 1 == 0])
上面有很多详细的解决方案,但我同意这是最简洁的。您甚至可以提取 ax.get_xticks() 的长度并将切片频率除以该长度除以所需的刻度数。
我认为这是最好的答案。大多数其他答案太复杂,难以应用/概括。谢谢!
它只能减少棍子的数量,而问题(以及我如何找到它的目标)是增加它。
C
Community

这有点老套,但到目前为止,我发现这样做是最干净/最容易理解的示例。这是来自这里的答案:

Cleanest way to hide every nth tick label in matplotlib colorbar?

for label in ax.get_xticklabels()[::2]:
    label.set_visible(False)

然后,您可以根据您想要的密度遍历标签,将它们设置为可见或不可见。

编辑:请注意,有时 matplotlib 设置标签 == '',因此它可能看起来像一个标签不存在,而实际上它是并且只是没有显示任何东西。为确保您循环浏览实际可见的标签,您可以尝试:

visible_labels = [lab for lab in ax.get_xticklabels() if lab.get_visible() is True and lab.get_text() != '']
plt.setp(visible_labels[::2], visible=False)

这是最简单和通用的解决方案。一个微小的调整:通常 ax.get_xticklabels()[1::2] 是要隐藏的标签。
这不适用于 matplotlib.finance.candlestick2
@BCR 可能是某些 xticklabels 刚刚设置为 '' ,因此当您遍历它们时,您正在制作空的 xticklabels 不可见(这对可视化没有影响,但可能意味着您'不拉正确的标签)。您可以尝试:vis_labels = [label for label in ax.get_xticklabels() if label.get_visible() is True]; plt.setp(vis_labels[::2], visible==False)
G
Gary Steele

如果您只想将间距设置为具有最少样板的简单单行:

plt.gca().xaxis.set_major_locator(plt.MultipleLocator(1))

也适用于小蜱:

plt.gca().xaxis.set_minor_locator(plt.MultipleLocator(1))

有点满口,但非常紧凑


值得一提的是参数 plt.MultipleLocator(arg) 是滴答间隔。因此,如果您希望您的刻度彼此相距 5 个单位,只需使用 plt.MultipleLocator(5)。否则最喜欢这个解决方案。谢谢!
T
Tompa

这是一个古老的话题,但我时不时地偶然发现这个并制作了这个功能。非常方便:

import matplotlib.pyplot as pp
import numpy as np

def resadjust(ax, xres=None, yres=None):
    """
    Send in an axis and I fix the resolution as desired.
    """

    if xres:
        start, stop = ax.get_xlim()
        ticks = np.arange(start, stop + xres, xres)
        ax.set_xticks(ticks)
    if yres:
        start, stop = ax.get_ylim()
        ticks = np.arange(start, stop + yres, yres)
        ax.set_yticks(ticks)

像这样控制刻度的一个警告是,在添加一行后,人们不再享受最大比例的交互式自动更新。然后做

gca().set_ylim(top=new_top) # for example

并再次运行 resadjust 功能。


D
Deninhos

我开发了一个不优雅的解决方案。考虑我们有 X 轴以及 X 中每个点的标签列表。

import matplotlib.pyplot as plt

x = [0,1,2,3,4,5]
y = [10,20,15,18,7,19]
xlabels = ['jan','feb','mar','apr','may','jun']
xlabelsnew = []
for i in xlabels:
    if i not in ['feb','jun']:
        i = ' '
        xlabelsnew.append(i)
    else:
        xlabelsnew.append(i)
plt.plot(x,y)
plt.xticks(range(0,len(x)),xlabels,rotation=45)
plt.show()
plt.plot(x,y)
plt.xticks(range(0,len(x)),xlabelsnew,rotation=45)
plt.show()

G
Greenstick

纯 Python 实现

下面是所需功能的纯 python 实现,它处理具有正、负或混合值的任何数字系列(int 或 float),并允许用户指定所需的步长:

import math

def computeTicks (x, step = 5):
    """
    Computes domain with given step encompassing series x
    @ params
    x    - Required - A list-like object of integers or floats
    step - Optional - Tick frequency
    """
    xMax, xMin = math.ceil(max(x)), math.floor(min(x))
    dMax, dMin = xMax + abs((xMax % step) - step) + (step if (xMax % step != 0) else 0), xMin - abs((xMin % step))
    return range(dMin, dMax, step)

样本输出

# Negative to Positive
series = [-2, 18, 24, 29, 43]
print(list(computeTicks(series)))

[-5, 0, 5, 10, 15, 20, 25, 30, 35, 40, 45]

# Negative to 0
series = [-30, -14, -10, -9, -3, 0]
print(list(computeTicks(series)))

[-30, -25, -20, -15, -10, -5, 0]

# 0 to Positive
series = [19, 23, 24, 27]
print(list(computeTicks(series)))

[15, 20, 25, 30]

# Floats
series = [1.8, 12.0, 21.2]
print(list(computeTicks(series)))

[0, 5, 10, 15, 20, 25]

# Step – 100
series = [118.3, 293.2, 768.1]
print(list(computeTicks(series, step = 100)))

[100, 200, 300, 400, 500, 600, 700, 800]

示例使用

import matplotlib.pyplot as plt

x = [0,5,9,10,15]
y = [0,1,2,3,4]
plt.plot(x,y)
plt.xticks(computeTicks(x))
plt.show()

https://i.stack.imgur.com/imxIG.png

请注意,x 轴的整数值均以 5 为间隔均匀分布,而 y 轴的间隔不同(matplotlib 默认行为,因为未指定刻度)。


J
Johnny V

通用的一个班轮,仅进口 Numpy:

ax.set_xticks(np.arange(min(x),max(x),1))

在问题的上下文中设置:

import numpy as np
import matplotlib.pyplot as plt 
fig, ax = plt.subplots()
x = [0,5,9,10,15]
y = [0,1,2,3,4]
ax.plot(x,y)
ax.set_xticks(np.arange(min(x),max(x),1))
plt.show()

这个怎么运作:

fig, ax = plt.subplots() 给出了包含轴的 ax 对象。 np.arange(min(x),max(x),1) 给出从 x 的最小值到 x 的最大值的区间为 1 的数组。这是我们想要的新的 x 刻度。 ax.set_xticks() 更改 ax 对象上的刻度。


那么,你能解释一下代码吗?
我已经重写它以使其更清晰。希望有帮助
B
BartoszKP
xmarks=[i for i in range(1,length+1,1)]

plt.xticks(xmarks)

这对我有用

如果您想要 [1,5](包括 1 和 5)之间的刻度,则替换

length = 5

仅供参考,您可以简单地写 xmarks = range(1, length+1, 1)。很确定列表理解是多余的。
k
kmario23

由于上述解决方案没有一个适用于我的用例,因此我在这里提供了一个使用 None(双关语!)的解决方案,它可以适应各种场景。

这是一段代码示例,它会在 XY 轴上产生杂乱的刻度。

# Note the super cluttered ticks on both X and Y axis.

# inputs
x = np.arange(1, 101)
y = x * np.log(x) 

fig = plt.figure()     # create figure
ax = fig.add_subplot(111)
ax.plot(x, y)
ax.set_xticks(x)        # set xtick values
ax.set_yticks(y)        # set ytick values

plt.show()

现在,我们用一个新图来清理混乱,该图仅将 x 和 y 轴上的一组稀疏值显示为刻度。

# inputs
x = np.arange(1, 101)
y = x * np.log(x)

fig = plt.figure()       # create figure
ax = fig.add_subplot(111)
ax.plot(x, y)

ax.set_xticks(x)
ax.set_yticks(y)

# which values need to be shown?
# here, we show every third value from `x` and `y`
show_every = 3

sparse_xticks = [None] * x.shape[0]
sparse_xticks[::show_every] = x[::show_every]

sparse_yticks = [None] * y.shape[0]
sparse_yticks[::show_every] = y[::show_every]

ax.set_xticklabels(sparse_xticks, fontsize=6)   # set sparse xtick values
ax.set_yticklabels(sparse_yticks, fontsize=6)   # set sparse ytick values

plt.show()

根据用例,只需更改 show_every 并将其用于对 X 或 Y 或两个轴的刻度值进行采样,即可调整上述代码。

如果这种基于步长的解决方案不适合,那么如果需要的话,还可以不定期地填充 sparse_xtickssparse_yticks 的值。


A
Ahmad

您可以遍历标签并显示或隐藏您想要的标签:

   for i, label in enumerate(ax.get_xticklabels()):
        if i % interval != 0:
            label.set_visible(False)