二十五、自定义图例

在这篇 Matplotlib 教程中,我们将讨论自定义图例。 我们已经介绍了添加图例的基础知识

图例的主要问题通常是图例阻碍了数据的展示。 这里有几个选项。 一个选项是将图例放在轴域外,但是我们在这里有多个子图,这是非常困难的。 相反,我们将使图例稍微小一点,然后应用一个透明度。

首先,为了创建一个图例,我们需要向我们的数据添加我们想要显示在图例上的标签。

  1. ax1.plot_date(date[-start:],h_l[-start:],'-', label='H-L')
  2. ...
  3. ax2v.plot([],[], color='#0079a3', alpha=0.4, label='Volume')
  4. ...
  5. ax3.plot(date[-start:], ma1[-start:], linewidth=1, label=(str(MA1)+'MA'))
  6. ax3.plot(date[-start:], ma2[-start:], linewidth=1, label=(str(MA2)+'MA'))

请注意,我们通过创建空行为交易量添加了标签。 请记住,我们不能对任何填充应用标签,所以这就是我们添加这个空行的原因。

现在,我们可以在右下角添加图例,通过在plt.show()之前执行以下操作:

  1. ax1.legend()
  2. ax2v.legend()
  3. ax3.legend()

会生成:

二十五、自定义图例 - 图1

所以,我们可以看到,图例还是占用了一些位置。 让我们更改位置,大小并添加透明度:

  1. ax1.legend()
  2. leg = ax1.legend(loc=9, ncol=2,prop={'size':11})
  3. leg.get_frame().set_alpha(0.4)
  4. ax2v.legend()
  5. leg = ax2v.legend(loc=9, ncol=2,prop={'size':11})
  6. leg.get_frame().set_alpha(0.4)
  7. ax3.legend()
  8. leg = ax3.legend(loc=9, ncol=2,prop={'size':11})
  9. leg.get_frame().set_alpha(0.4)

所有的图例位于位置 9(上中间)。 有很多地方可放置图例,我们可以为参数传入不同的位置号码,来看看它们都位于哪里。 ncol参数允许我们指定图例中的列数。 这里只有一列,如果图例中有 2 个项目,他们将堆叠在一列中。 最后,我们将尺寸规定为更小。 之后,我们对整个图例应用0.4的透明度。

现在我们的结果为:

二十五、自定义图例 - 图2

完整的代码为:

  1. import matplotlib.pyplot as plt
  2. import matplotlib.dates as mdates
  3. import matplotlib.ticker as mticker
  4. from matplotlib.finance import candlestick_ohlc
  5. from matplotlib import style
  6. import numpy as np
  7. import urllib
  8. import datetime as dt
  9. style.use('fivethirtyeight')
  10. print(plt.style.available)
  11. print(plt.__file__)
  12. MA1 = 10
  13. MA2 = 30
  14. def moving_average(values, window):
  15. weights = np.repeat(1.0, window)/window
  16. smas = np.convolve(values, weights, 'valid')
  17. return smas
  18. def high_minus_low(highs, lows):
  19. return highs-lows
  20. def bytespdate2num(fmt, encoding='utf-8'):
  21. strconverter = mdates.strpdate2num(fmt)
  22. def bytesconverter(b):
  23. s = b.decode(encoding)
  24. return strconverter(s)
  25. return bytesconverter
  26. def graph_data(stock):
  27. fig = plt.figure(facecolor='#f0f0f0')
  28. ax1 = plt.subplot2grid((6,1), (0,0), rowspan=1, colspan=1)
  29. plt.title(stock)
  30. plt.ylabel('H-L')
  31. ax2 = plt.subplot2grid((6,1), (1,0), rowspan=4, colspan=1, sharex=ax1)
  32. plt.ylabel('Price')
  33. ax2v = ax2.twinx()
  34. ax3 = plt.subplot2grid((6,1), (5,0), rowspan=1, colspan=1, sharex=ax1)
  35. plt.ylabel('MAvgs')
  36. stock_price_url = 'http://chartapi.finance.yahoo.com/instrument/1.0/'+stock+'/chartdata;type=quote;range=1y/csv'
  37. source_code = urllib.request.urlopen(stock_price_url).read().decode()
  38. stock_data = []
  39. split_source = source_code.split('\n')
  40. for line in split_source:
  41. split_line = line.split(',')
  42. if len(split_line) == 6:
  43. if 'values' not in line and 'labels' not in line:
  44. stock_data.append(line)
  45. date, closep, highp, lowp, openp, volume = np.loadtxt(stock_data,
  46. delimiter=',',
  47. unpack=True,
  48. converters={0: bytespdate2num('%Y%m%d')})
  49. x = 0
  50. y = len(date)
  51. ohlc = []
  52. while x < y:
  53. append_me = date[x], openp[x], highp[x], lowp[x], closep[x], volume[x]
  54. ohlc.append(append_me)
  55. x+=1
  56. ma1 = moving_average(closep,MA1)
  57. ma2 = moving_average(closep,MA2)
  58. start = len(date[MA2-1:])
  59. h_l = list(map(high_minus_low, highp, lowp))
  60. ax1.plot_date(date[-start:],h_l[-start:],'-', label='H-L')
  61. ax1.yaxis.set_major_locator(mticker.MaxNLocator(nbins=4, prune='lower'))
  62. candlestick_ohlc(ax2, ohlc[-start:], width=0.4, colorup='#77d879', colordown='#db3f3f')
  63. ax2.yaxis.set_major_locator(mticker.MaxNLocator(nbins=7, prune='upper'))
  64. ax2.grid(True)
  65. bbox_props = dict(boxstyle='round',fc='w', ec='k',lw=1)
  66. ax2.annotate(str(closep[-1]), (date[-1], closep[-1]),
  67. xytext = (date[-1]+4, closep[-1]), bbox=bbox_props)
  68. ## # Annotation example with arrow
  69. ## ax2.annotate('Bad News!',(date[11],highp[11]),
  70. ## xytext=(0.8, 0.9), textcoords='axes fraction',
  71. ## arrowprops = dict(facecolor='grey',color='grey'))
  72. ##
  73. ##
  74. ## # Font dict example
  75. ## font_dict = {'family':'serif',
  76. ## 'color':'darkred',
  77. ## 'size':15}
  78. ## # Hard coded text
  79. ## ax2.text(date[10], closep[1],'Text Example', fontdict=font_dict)
  80. ax2v.plot([],[], color='#0079a3', alpha=0.4, label='Volume')
  81. ax2v.fill_between(date[-start:],0, volume[-start:], facecolor='#0079a3', alpha=0.4)
  82. ax2v.axes.yaxis.set_ticklabels([])
  83. ax2v.grid(False)
  84. ax2v.set_ylim(0, 3*volume.max())
  85. ax3.plot(date[-start:], ma1[-start:], linewidth=1, label=(str(MA1)+'MA'))
  86. ax3.plot(date[-start:], ma2[-start:], linewidth=1, label=(str(MA2)+'MA'))
  87. ax3.fill_between(date[-start:], ma2[-start:], ma1[-start:],
  88. where=(ma1[-start:] < ma2[-start:]),
  89. facecolor='r', edgecolor='r', alpha=0.5)
  90. ax3.fill_between(date[-start:], ma2[-start:], ma1[-start:],
  91. where=(ma1[-start:] > ma2[-start:]),
  92. facecolor='g', edgecolor='g', alpha=0.5)
  93. ax3.xaxis.set_major_formatter(mdates.DateFormatter('%Y-%m-%d'))
  94. ax3.xaxis.set_major_locator(mticker.MaxNLocator(10))
  95. ax3.yaxis.set_major_locator(mticker.MaxNLocator(nbins=4, prune='upper'))
  96. for label in ax3.xaxis.get_ticklabels():
  97. label.set_rotation(45)
  98. plt.setp(ax1.get_xticklabels(), visible=False)
  99. plt.setp(ax2.get_xticklabels(), visible=False)
  100. plt.subplots_adjust(left=0.11, bottom=0.24, right=0.90, top=0.90, wspace=0.2, hspace=0)
  101. ax1.legend()
  102. leg = ax1.legend(loc=9, ncol=2,prop={'size':11})
  103. leg.get_frame().set_alpha(0.4)
  104. ax2v.legend()
  105. leg = ax2v.legend(loc=9, ncol=2,prop={'size':11})
  106. leg.get_frame().set_alpha(0.4)
  107. ax3.legend()
  108. leg = ax3.legend(loc=9, ncol=2,prop={'size':11})
  109. leg.get_frame().set_alpha(0.4)
  110. plt.show()
  111. fig.savefig('google.png', facecolor=fig.get_facecolor())
  112. graph_data('GOOG')

现在我们可以看到图例,但也看到了图例下的任何信息。 还要注意额外函数fig.savefig。 这是自动保存图形的图像的方式。 我们还可以设置所保存的图形的前景色,使背景不是白色的,如我们的例子所示。

这就是目前为止,我想要显示的典型 Matplotlib 图表。 接下来,我们将涉及Basemap,它是一个 Matplotlib 扩展,用于绘制地理位置,然后我打算讲解 Matplotlib 中的 3D 图形。