强基初中数学&学Python——第260课 数字和数学第三方模块Matplotlib之七:封装端-可组合循环器

Composable cycles

版本:0.11.0

日期:20211110

Docs

https://matplotlib.org/cycler

PyPI

https://pypi.python.org/pypi/Cycler

GitHub

https://github.com/matplotlib/cycler

 

循环器(cyclerAPI

cycler(*args, **kwargs)

从单个位置参数、一对位置参数或关键字参数的组合创建新的Cycler对象的函数。

Cycler(left[, right, op])

可组合循环(Cycler对象)构造器

concat(left, right)

连接Cyclers,就像使用itertools.chain链接一样。

  cycler的公共API由一个类Cycler、一个工厂函数cycler()和一个连接函数concat()组成。工厂函数提供了一个简单的接口来创建基本Cycler对象,而类负责组合和迭代逻辑。

Cycler:

https://matplotlib.org/cycler/generated/cycler.Cycler.html#cycler.Cycler

循环器(Cycler)用法

基础(Base

  可以使用单个条目Cycler对象轻松地在单个样式上循环。要创建Cycler,请使用cycler()函数,它将key()/style(样式)/kwarg(关键字参数)链接到一系列值。key必须是可散列的(因为它最终将用作dict中的键(key))。

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

ipython
In [1]: from __future__ import print_function
In [2]: from cycler import cycler
In [3]: color_cycle = cycler(color=['r', 'g', 'b'])
In [4]: color_cycleOut[4]: cycler('color', ['r', 'g', 'b'])

  Cycler知道它的长度和键(一个集合set):

· 

· 

· 

· 

· 

In [5]: len(color_cycle)Out[5]: 3
In [6]: color_cycle.keysOut[6]: {'color'}

  在该对象上迭代将产生一系列以标签(Cyclerkeys)为键的dict对象:

· 

· 

· 

· 

· 

· 

In [7]: for v in color_cycle:   ...:     print(v)   ...:{'color': 'r'}{'color': 'g'}{'color': 'b'}

  Cycler对象可以作为参数传递给cycler(),后者返回一个带有新标签但值相同的新Cycler

· 

· 

In [8]: cycler(ec=color_cycle)Out[8]: cycler('ec', ['r', 'g', 'b'])

  Cycler上迭代会导致有限的条目列表,要获得无限循环,请调用Cycler对象(a-la生成器)

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

In [9]: cc = color_cycle()
In [10]: for j, c in zip(range(5),  cc):    ...:     print(j, c)    ...:0 {'color': 'r'}1 {'color': 'g'}2 {'color': 'b'}3 {'color': 'r'}4 {'color': 'g'}

 

组合方式(Composition

  单个Cycler同样可以被单个for循环替换。Cycler对象的强大之处在于,它们可以组合起来,轻松创建复杂的多键循环。

加法(Addition

  具有不同键的等长Cycler可以相加,以获得两个循环的内积

· 

· 

· 

In [11]: lw_cycle = cycler(lw=range(1, 4))
In [12]: wc = lw_cycle + color_cycle

  结果具有相同的长度,并且具有两个输入Cycler的键的并集的键。

· 

· 

· 

· 

· 

In [13]: len(wc)Out[13]: 3
In [14]: wc.keysOut[14]: {'color', 'lw'}

  对结果进行迭代是两个输入Cyclerzip(每一个项含有对应2个输入的键值对)。

· 

· 

· 

· 

· 

· 

In [15]: for s in wc:    ...:     print(s)    ...:{'lw': 1, 'color': 'r'}{'lw': 2, 'color': 'g'}{'lw': 3, 'color': 'b'}

  与算术一样,加法是可交换的。

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

In [16]: lw_c = lw_cycle + color_cycle
In [17]: c_lw = color_cycle + lw_cycle
In [18]: for j, (a, b) in enumerate(zip(lw_c, c_lw)):    ...:     print('({j}) A: {A!r} B: {B!r}'.format(j=j, A=a, B=b))    ...:(0) A: {'lw': 1, 'color': 'r'} B: {'color': 'r', 'lw': 1}(1) A: {'lw': 2, 'color': 'g'} B: {'color': 'g', 'lw': 2}(2) A: {'lw': 3, 'color': 'b'} B: {'color': 'b', 'lw': 3}

  为了方便起见,cycler()函数可以有多个键值对,并通过相加将它们自动组合成一个Cycler

· 

· 

· 

· 

· 

· 

· 

· 

In [19]: wc = cycler(c=['r', 'g', 'b'], lw=range(3))
In [20]: for s in wc:    ...:     print(s)    ...:{'c': 'r', 'lw': 0}{'c': 'g', 'lw': 1}{'c': 'b', 'lw': 2}

 

乘法(Multiplication

  任何一对Cycler都可以相乘。

· 

· 

· 

In [21]: m_cycle = cycler(marker=['s', 'o'])
In [22]: m_c = m_cycle * color_cycle

  它给出两个循环的外积(与itertools.product()相同)。

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

In [23]: len(m_c)Out[23]: 6
In [24]: m_c.keysOut[24]: {'color', 'marker'}
In [25]: for s in m_c:    ...:     print(s)    ...:{'marker': 's', 'color': 'r'}{'marker': 's', 'color': 'g'}{'marker': 's', 'color': 'b'}{'marker': 'o', 'color': 'r'}{'marker': 'o', 'color': 'g'}{'marker': 'o', 'color': 'b'}

  注意,与加法不同,乘法不可交换(如矩阵)。结果的项目集合虽然一样,但对应的项目可能不一样。

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

In [26]: c_m = color_cycle * m_cycle
In [27]: for j, (a, b) in enumerate(zip(c_m, m_c)):    ...:     print('({j}) A: {A!r} B: {B!r}'.format(j=j, A=a, B=b))    ...:(0) A: {'color': 'r', 'marker': 's'} B: {'marker': 's', 'color': 'r'}(1) A: {'color': 'r', 'marker': 'o'} B: {'marker': 's', 'color': 'g'}(2) A: {'color': 'g', 'marker': 's'} B: {'marker': 's', 'color': 'b'}(3) A: {'color': 'g', 'marker': 'o'} B: {'marker': 'o', 'color': 'r'}(4) A: {'color': 'b', 'marker': 's'} B: {'marker': 'o', 'color': 'g'}(5) A: {'color': 'b', 'marker': 'o'} B: {'marker': 'o', 'color': 'b'}

 

整数相乘(Integer Multiplication

  Cycler也可以乘以整数值以增加长度。

· 

· 

· 

· 

· 

In [28]: color_cycle * 2Out[28]: cycler('color', ['r', 'g', 'b', 'r', 'g', 'b'])
In [29]: 2 * color_cycleOut[29]: cycler('color', ['r', 'g', 'b', 'r', 'g', 'b'])

 

连接(Concatenation

  可以通过Cycler.concat()方法连接Cycler对象:

· 

· 

In [30]: color_cycle.concat(color_cycle)Out[30]: cycler('color', ['r', 'g', 'b', 'r', 'g', 'b'])

  或顶级concat()函数:

· 

· 

· 

· 

In [31]: from cycler import concat
In [32]: concat(color_cycle, color_cycle)Out[32]: cycler('color', ['r', 'g', 'b', 'r', 'g', 'b'])

 

切片(Slicing

  可以使用slice(切片)对象切片循环。

· 

· 

· 

· 

· 

· 

· 

· 

In [33]: color_cycle[::-1]Out[33]: cycler('color', ['b', 'g', 'r'])
In [34]: color_cycle[:2]Out[34]: cycler('color', ['r', 'g'])
In [35]: color_cycle[1:]Out[35]: cycler('color', ['g', 'b'])

  以返回循环的子集作为新的Cycler

检查Cycler

  要检查转置的Cycler的值,请使用Cycler.by_key方法:

· 

· 

· 

· 

In [36]: c_m.by_key()Out[36]:{'marker': ['s', 'o', 's', 'o', 's', 'o'], 'color': ['r', 'r', 'g', 'g', 'b', 'b']}

  dict可以被改变,并用于创建具有更新值的新Cycler

· 

· 

· 

· 

· 

· 

In [37]: bk = c_m.by_key()
In [38]: bk['color'] = ['green'] * len(c_m)
In [39]: cycler(**bk)Out[39]: (cycler('marker', ['s', 'o', 's', 'o', 's', 'o']) + cycler('color', ['green', 'green', 'green', 'green', 'green', 'green']))

 

示例

  我们可以使用Cycler实例在一个或多个kwarg上循环以绘制(plot):

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

from cycler import cyclerfrom itertools import cycle
fig, (ax1, ax2) = plt.subplots(1, 2, tight_layout=True,                               figsize=(8, 4))x = np.arange(10)
color_cycle = cycler(c=['r', 'g', 'b'])
for i, sty in enumerate(color_cycle):   ax1.plot(x, x*(i+1), **sty)

for i, sty in zip(range(1, 5), cycle(color_cycle)):   ax2.plot(x, x*i, **sty)

  完整代码:

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

ipython%matplotlib tkimport matplotlib.pyplot as pltimport numpy as npfrom cycler import cyclerfrom itertools import cycle
fig, (ax1, ax2) = plt.subplots(1, 2, tight_layout=True,                               figsize=(8, 4))x = np.arange(10)
color_cycle = cycler(c=['r', 'g', 'b'])
for i, sty in enumerate(color_cycle):   ax1.plot(x, x*(i+1), **sty)

for i, sty in zip(range(1, 5), cycle(color_cycle)):   ax2.plot(x, x*i, **sty)

 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

from cycler import cyclerfrom itertools import cycle
fig, (ax1, ax2) = plt.subplots(1, 2, tight_layout=True,                               figsize=(8, 4))x = np.arange(10)
color_cycle = cycler(c=['r', 'g', 'b'])ls_cycle = cycler('ls', ['-', '--'])lw_cycle = cycler('lw', range(1, 4))
sty_cycle = ls_cycle * (color_cycle + lw_cycle)
for i, sty in enumerate(sty_cycle):   ax1.plot(x, x*(i+1), **sty)
sty_cycle = (color_cycle + lw_cycle) * ls_cycle
for i, sty in enumerate(sty_cycle):   ax2.plot(x, x*(i+1), **sty)

  完整代码:

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

ipython%matplotlib tkimport matplotlib.pyplot as pltimport numpy as npfrom cycler import cyclerfrom itertools import cycle
fig, (ax1, ax2) = plt.subplots(1, 2, tight_layout=True,                               figsize=(8, 4))x = np.arange(10)
color_cycle = cycler(c=['r', 'g', 'b'])ls_cycle = cycler('ls', ['-', '--'])lw_cycle = cycler('lw', range(1, 4))
sty_cycle = ls_cycle * (color_cycle + lw_cycle)
for i, sty in enumerate(sty_cycle):   ax1.plot(x, x*(i+1), **sty)
sty_cycle = (color_cycle + lw_cycle) * ls_cycle
for i, sty in enumerate(sty_cycle):   ax2.plot(x, x*(i+1), **sty)

 

 

持续性循环

  通过字典查找将给定标签与样式相关联,并动态生成该映射可能很有用。这可以使用defaultdict轻松完成。

· 

· 

· 

· 

· 

In [40]: from cycler import cycler as cy
In [41]: from collections import defaultdict
In [42]: cyl = cy('c', 'rgb') + cy('lw', range(1, 4))

  获取有限的样式集。

· 

· 

· 

In [43]: finite_cy_iter = iter(cyl)
In [44]: dd_finite = defaultdict(lambda : next(finite_cy_iter))

  或重复

· 

· 

· 

In [45]: loop_cy_iter = cyl()
In [46]: dd_loop = defaultdict(lambda : next(loop_cy_iter))

  这在绘制既有分类又有标签的复杂数据时非常有用。

· 

· 

· 

· 

finite_cy_iter = iter(cyl)styles = defaultdict(lambda : next(finite_cy_iter))for group, label, data in DataSet:    ax.plot(data, label=label, **styles[group])

  这将导致具有相同组的每个数据以相同样式绘制。

异常

  如果将不等长度的Cycler相加,则会引发ValueError

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

In [47]: cycler(c=['r', 'g', 'b']) + cycler(ls=['-', '--'])---------------------------------------------------------------------------ValueError                                Traceback (most recent call last)Cell In[47], line 1----> 1 cycler(c=['r', 'g', 'b']) + cycler(ls=['-', '--'])
File D:\5xstar\Python\Lib\site-packages\cycler.py:259, in Cycler.__add__(self, other)    251 """    252 Pair-wise combine two equal length cyclers (zip).    253   (...)    256 other : Cycler    257 """    258 if len(self) != len(other):--> 259     raise ValueError("Can only add equal length cycles, "    260                      f"not {len(self)} and {len(other)}")    261 return Cycler(self, other, zip)
ValueError: Can only add equal length cycles, not 3 and 2

  或者如果组成了具有重叠键的两个循环。

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

In [48]: color_cycle = cycler(c=['r', 'g', 'b'])
In [49]: color_cycle + color_cycle---------------------------------------------------------------------------ValueError                                Traceback (most recent call last)Cell In[49], line 1----> 1 color_cycle + color_cycle
File D:\5xstar\Python\Lib\site-packages\cycler.py:261, in Cycler.__add__(self, other)    258 if len(self) != len(other):    259     raise ValueError("Can only add equal length cycles, "    260                      f"not {len(self)} and {len(other)}")--> 261 return Cycler(self, other, zip)
File D:\5xstar\Python\Lib\site-packages\cycler.py:159, in Cycler.__init__(self, left, right, op)    156 else:    157     self._right = None--> 159 self._keys = _process_keys(self._left, self._right)    160 self._op = op
File D:\5xstar\Python\Lib\site-packages\cycler.py:71, in _process_keys(left, right)     69 r_key = set(r_peek.keys())     70 if l_key & r_key:---> 71     raise ValueError("Can not compose overlapping cycles")     72 return l_key | r_key
ValueError: Can not compose overlapping cycles

 

动机(Motivation

  当绘制多条线时,通常希望能够在一个或多个艺术家样式之间循环。对于不需要太多麻烦就能完成的简单情况:

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

fig, ax = plt.subplots(tight_layout=True)x = np.linspace(0, 2*np.pi, 1024)
for i, (lw, c) in enumerate(zip(range(4), ['r', 'g', 'b', 'k'])):   ax.plot(x, np.sin(x - i * np.pi / 4),           label=r'$\phi = {{{0}}} \pi / 4$'.format(i),           lw=lw + 1,           c=c)
ax.set_xlim([0, 2*np.pi])ax.set_title(r'$y=\sin(\theta + \phi)$')ax.set_ylabel(r'[arb]')ax.set_xlabel(r'$\theta$ [rad]')
ax.legend(loc=0)

  完整代码:

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

ipython%matplotlib tkimport matplotlib.pyplot as pltimport numpy as npfig, ax = plt.subplots(tight_layout=True)x = np.linspace(0, 2*np.pi, 1024)
for i, (lw, c) in enumerate(zip(range(4), ['r', 'g', 'b', 'k'])):   ax.plot(x, np.sin(x - i * np.pi / 4),           label=r'$\phi = {{{0}}} \pi / 4$'.format(i),           lw=lw + 1,           c=c)
ax.set_xlim([0, 2*np.pi])ax.set_title(r'$y=\sin(\theta + \phi)$')ax.set_ylabel(r'[arb]')ax.set_xlabel(r'$\theta$ [rad]')
ax.legend(loc=0)

 

  再者,做复杂一些的事情:

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

fig, ax = plt.subplots(tight_layout=True)x = np.linspace(0, 2*np.pi, 1024)
for i, (lw, c) in enumerate(zip(range(4), ['r', 'g', 'b', 'k'])):   if i % 2:       ls = '-'   else:       ls = '--'   ax.plot(x, np.sin(x - i * np.pi / 4),           label=r'$\phi = {{{0}}} \pi / 4$'.format(i),           lw=lw + 1,           c=c,           ls=ls)
ax.set_xlim([0, 2*np.pi])ax.set_title(r'$y=\sin(\theta + \phi)$')ax.set_ylabel(r'[arb]')ax.set_xlabel(r'$\theta$ [rad]')
ax.legend(loc=0)

  完整代码:

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

· 

ipython%matplotlib tkimport matplotlib.pyplot as pltimport numpy as npfig, ax = plt.subplots(tight_layout=True)x = np.linspace(0, 2*np.pi, 1024)
for i, (lw, c) in enumerate(zip(range(4), ['r', 'g', 'b', 'k'])):   if i % 2:       ls = '-'   else:       ls = '--'   ax.plot(x, np.sin(x - i * np.pi / 4),           label=r'$\phi = {{{0}}} \pi / 4$'.format(i),           lw=lw + 1,           c=c,           ls=ls)
ax.set_xlim([0, 2*np.pi])ax.set_title(r'$y=\sin(\theta + \phi)$')ax.set_ylabel(r'[arb]')ax.set_xlabel(r'$\theta$ [rad]')
ax.legend(loc=0)

 

  然而,绘图逻辑可以变得非常复杂,上面简单的循环方法已无法解决参数问题。为了解决这一问题,并允许在任意关键字参数(kwargs)上轻松循环,开发了Cycler类,这是一个可组合的关键字参数(kwargs)迭代器。

文中链接

cycler():

https://matplotlib.org/cycler/generated/cycler.cycler.html#cycler.cycler

Cycler:

https://matplotlib.org/cycler/generated/cycler.Cycler.html#cycler.Cycler

concat():

https://matplotlib.org/cycler/generated/cycler.concat.html#cycler.concat

itertools.chain:

https://docs.python.org/3/library/itertools.html#itertools.chain

dict:

https://docs.python.org/3/library/stdtypes.html#dict

itertools.product():

https://docs.python.org/3/library/itertools.html#itertools.product

Cycler.concat():

https://matplotlib.org/cycler/generated/cycler.Cycler.html#cycler.Cycler.concat

slice:

https://docs.python.org/3/library/functions.html#slice

Cycler.by_key:

https://matplotlib.org/cycler/generated/cycler.Cycler.html#cycler.Cycler.by_key

plot:

https://matplotlib.org/api/_as_gen/matplotlib.axes.Axes.plot.html#matplotlib.axes.Axes.plot

defaultdict:

https://docs.python.org/3/library/collections.html#collections.defaultdict

ValueError:

https://docs.python.org/3/library/exceptions.html#ValueError