如何突出显示特定的 x 值范围

2025-03-05 09:17:00
admin
原创
85
摘要:问题描述:我正在为一个项目制作历史股票数据的可视化,我想突出显示下跌区域。例如,当股票经历大幅下跌时,我想用红色区域突出显示它。我可以自动完成此操作吗,还是必须绘制一个矩形或类似的东西?解决方案 1:看一下axvspan(以及用于突出显示 y 轴区域的 axhspan)。import matplotlib.p...

问题描述:

我正在为一个项目制作历史股票数据的可视化,我想突出显示下跌区域。例如,当股票经历大幅下跌时,我想用红色区域突出显示它。

我可以自动完成此操作吗,还是必须绘制一个矩形或类似的东西?


解决方案 1:

看一下axvspan(以及用于突出显示 y 轴区域的 axhspan)。

import matplotlib.pyplot as plt

plt.plot(range(10))
plt.axvspan(3, 6, color='red', alpha=0.5)
plt.show()

在此处输入图片描述

如果您使用日期,则需要将最小和最大 x 值转换为 matplotlib 日期。用于matplotlib.dates.date2num对象datetimematplotlib.dates.datestr2num各种字符串时间戳。

import matplotlib.pyplot as plt
import matplotlib.dates as mdates
import datetime as dt

t = mdates.drange(dt.datetime(2011, 10, 15), dt.datetime(2011, 11, 27),
                  dt.timedelta(hours=2))
y = np.sin(t)

fig, ax = plt.subplots()
ax.plot_date(t, y, 'b-')
ax.axvspan(*mdates.datestr2num(['10/27/2011', '11/2/2011']), color='red', alpha=0.5)
fig.autofmt_xdate()
plt.show()

在此处输入图片描述

解决方案 2:

这是一个用于axvspan绘制多个亮点的解决方案,其中每个亮点的限制是使用与峰值和谷值相对应的股票数据的指数来设置的。

股票数据通常包含不连续的时间变量,其中不包括周末和节假日。在处理每日股票价格时,在 matplotlib 或 pandas 中绘制它们会在周末和节假日的 x 轴上产生间隙。对于较长的日期范围和/或较小的数字(如本例中所示),这可能并不明显,但如果您放大,就会变得明显,这可能是您想要避免的事情。

这就是为什么我在这里分享一个完整的示例,其特点是:

  • 一个真实的样本数据集,其中包括基于使用pandas_market_calendarsDatetimeIndex导入的纽约证券交易所交易日历的不连续数据以及看起来像真实数据的虚假股票数据。

  • 用它创建的pandas 图通过use_index=False使用 x 轴的整数范围来消除周末和节假日的间隙。返回ax对象的使用方式避免了导入 matplotlib.pyplot(除非您需要plt.show)。

  • 使用 scipy.signal 函数自动检测整个日期范围内的亏损,find_peaks该函数返回绘制亮点所需的索引axvspan。以更正确的方式计算亏损需要明确定义什么算作亏损,并会导致更复杂的代码,这是另一个问题的主题。

  • 通过循环遍历时间戳创建的正确格式的刻度DatetimeIndex,因为所有方便的matplotlib.dates刻度定位器和格式化程序以及DatetimeIndex类似的属性.is_month_start在这种情况下无法使用。

创建示例数据集

import numpy as np                        # v 1.19.2
import pandas as pd                       # v 1.1.3
import pandas_market_calendars as mcal    # v 1.6.1
from scipy.signal import find_peaks       # v 1.5.2

# Create datetime index with a 'trading day end' frequency based on the New York Stock
# Exchange trading hours (end date is inclusive)
nyse = mcal.get_calendar('NYSE')
nyse_schedule = nyse.schedule(start_date='2019-10-01', end_date='2021-02-01')
nyse_dti = mcal.date_range(nyse_schedule, frequency='1D').tz_convert(nyse.tz.zone)

# Create sample of random data for daily stock closing price
rng = np.random.default_rng(seed=1234)  # random number generator
price = 100 + rng.normal(size=nyse_dti.size).cumsum()
df = pd.DataFrame(data=dict(price=price), index=nyse_dti)
df.head()

#                                    price
#   2019-10-01 16:00:00-04:00    98.396163
#   2019-10-02 16:00:00-04:00    98.460263
#   2019-10-03 16:00:00-04:00    99.201154
#   2019-10-04 16:00:00-04:00    99.353774
#   2019-10-07 16:00:00-04:00   100.217517

用正确格式的刻度标出亏损的高亮部分

# Plot stock price
ax = df['price'].plot(figsize=(10, 5), use_index=False, ylabel='Price')
ax.set_xlim(0, df.index.size-1)
ax.grid(axis='x', alpha=0.3)

# Highlight drawdowns using the indices of stock peaks and troughs: find peaks and 
# troughs based on signal analysis rather than an algorithm for drawdowns to keep
# example simple. Width and prominence have been handpicked for this example to work.
peaks, _ = find_peaks(df['price'], width=7, prominence=4)
troughs, _ = find_peaks(-df['price'], width=7, prominence=4)
for peak, trough in zip(peaks, troughs):
    ax.axvspan(peak, trough, facecolor='red', alpha=.2)

# Create and format monthly ticks
ticks = [idx for idx, timestamp in enumerate(df.index)
         if (timestamp.month != df.index[idx-1].month) | (idx == 0)]
ax.set_xticks(ticks)
labels = [tick.strftime('%b
%Y') if df.index[ticks[idx]].year
          != df.index[ticks[idx-1]].year else tick.strftime('%b')
          for idx, tick in enumerate(df.index[ticks])]
ax.set_xticklabels(labels)
ax.figure.autofmt_xdate(rotation=0, ha='center')

ax.set_title('Drawdowns are highlighted in red', pad=15, size=14);

亮点_亏损

为了完整起见,值得注意的是,您可以使用fill_between绘图函数实现完全相同的结果,尽管它需要多几行代码:

ax.set_ylim(*ax.get_ylim())  # remove top and bottom gaps with plot frame
drawdowns = np.repeat(False, df['price'].size)
for peak, trough in zip(peaks, troughs):
    drawdowns[np.arange(peak, trough+1)] = True
ax.fill_between(np.arange(df.index.size), *ax.get_ylim(), where=drawdowns,
                facecolor='red', alpha=.2)

您正在使用 matplotlib 的交互式界面,并希望在放大时显示动态刻度?那么您需要使用matplotlib.ticker模块中的定位器和格式化程序。例如,您可以像本例中一样保持主刻度固定,并在放大时添加动态次要刻度以显示一年中的几天或几周。您可以在此答案的末尾找到如何执行此操作的示例。

相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   4008  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   2751  
  本文介绍了以下10款项目管理软件工具:禅道项目管理软件、Freshdesk、ClickUp、nTask、Hubstaff、Plutio、Productive、Targa、Bonsai、Wrike。在当今快速变化的商业环境中,项目管理已成为企业成功的关键因素之一。然而,许多企业在项目管理过程中面临着诸多痛点,如任务分配不...
项目管理系统   86  
  本文介绍了以下10款项目管理软件工具:禅道项目管理软件、Monday、TeamGantt、Filestage、Chanty、Visor、Smartsheet、Productive、Quire、Planview。在当今快速变化的商业环境中,项目管理已成为企业成功的关键因素之一。然而,许多项目经理和团队在管理复杂项目时,常...
开源项目管理工具   97  
  本文介绍了以下10款项目管理软件工具:禅道项目管理软件、Smartsheet、GanttPRO、Backlog、Visor、ResourceGuru、Productive、Xebrio、Hive、Quire。在当今快节奏的商业环境中,项目管理已成为企业成功的关键因素之一。然而,许多企业在选择项目管理工具时常常面临困惑:...
项目管理系统   85  
热门文章
项目管理软件有哪些?
曾咪二维码

扫码咨询,免费领取项目管理大礼包!

云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用