Выравнивание координат тика по верхней и нижней оси - PullRequest
0 голосов
/ 08 февраля 2019

Я хотел бы построить фигуру, у которой на нижней оси х есть один набор тиков, а на верхней оси х другой набор тиков, который выровнен по нижним тикам.Конкретно в моем случае это партии и эпохи.Для каждых n точек партии (не обязательно отмеченных галочками) внизу я хочу отметку эпохи сверху.Рассмотрим этот пример:

import numpy as np
import matplotlib.pyplot as plt

batches = np.arange(1,101)
epoch_ends = batches[[(i*10)-1 for i in range(1,11)]]
accuracy = np.apply_along_axis(arr=batches, axis=0, func1d=lambda x : x/len(batches))
loss = np.apply_along_axis(arr=batches, axis=0, func1d=lambda x : 1 - (x/len(batches)))

fig, ax1 = plt.subplots( nrows=1, ncols=1 )
ax2 = ax1.twinx()
ax3 = ax1.twiny()

ax1.set_xlabel('batches')
ax1.set_xticks(np.arange(1, len(batches)+1, 9))
ax1.set_ylabel('accuracy')
ax1.grid()

ax2.set_ylabel('loss')
ax2.set_yticklabels(np.linspace(3, 10, 9))

ax3.set_xlabel('epochs')
ax3.set_xticks(epoch_ends)
ax3.set_xticklabels(range(1, len(epoch_ends)+1))

acc_plt = ax1.plot(batches, accuracy, color='red', label='acc')
loss_plt = ax2.plot(batches, loss, color='blue', label='loss')

lns = acc_plt+loss_plt
labs = [l.get_label() for l in lns]
ax1.legend(lns, labs, loc=2)

plt.show()

batches и epoch_ends соответственно выглядят следующим образом

[  1   2   3   4   5   6   7   8   9  10  11  12  13  14  15  16  17  18
  19  20  21  22  23  24  25  26  27  28  29  30  31  32  33  34  35  36
  37  38  39  40  41  42  43  44  45  46  47  48  49  50  51  52  53  54
  55  56  57  58  59  60  61  62  63  64  65  66  67  68  69  70  71  72
  73  74  75  76  77  78  79  80  81  82  83  84  85  86  87  88  89  90
  91  92  93  94  95  96  97  98  99 100]
[ 10  20  30  40  50  60  70  80  90 100]

Так что я бы хотел, чтобы тик эпохи 1 совпал с пакетом x-coorrdiante 10,2 с 20 и т. Д.

Но, как вы можете видеть на рисунке, они не совпадают.

enter image description here

Что делаютМне нужно изменить код, чтобы эта работа работала?

Ответы [ 2 ]

0 голосов
/ 08 февраля 2019

Это теряет контроль над тиками (вы не сказали, требовалось ли это или нет), но выстроит в линию две оси X, как вы и сказали: (добавлены новые строки)

import numpy as np
import matplotlib.pyplot as plt

batches = np.arange(1,101)
epoch_ends = batches[[(i*10)-1 for i in range(1,11)]]
accuracy = np.apply_along_axis(arr=batches, axis=0, func1d=lambda x : x/len(batches))
loss = np.apply_along_axis(arr=batches, axis=0, func1d=lambda x : 1 - (x/len(batches)))

fig, ax1 = plt.subplots( nrows=1, ncols=1 )
ax2 = ax1.twinx()
ax3 = ax1.twiny()

ax1.set_xlabel('batches')
#ax1.set_xticks(np.arange(1, len(batches)+1, 9))
ax1.set_xlim(0, 100)                               # Set xlim
ax1.set_ylabel('accuracy')
ax1.grid()

ax2.set_ylabel('loss')
ax2.set_yticklabels(np.linspace(3, 10, 9))

ax3.set_xlabel('epochs')
#ax3.set_xticks(epoch_ends)
#ax3.set_xticklabels(range(1, len(epoch_ends)+1))
ax3.set_xlim(0, 10)                                # Set xlim

acc_plt = ax1.plot(batches, accuracy, color='red', label='acc')
loss_plt = ax2.plot(batches, loss, color='blue', label='loss')

lns = acc_plt+loss_plt
labs = [l.get_label() for l in lns]
ax1.legend(lns, labs, loc=2)

plt.show()
0 голосов
/ 08 февраля 2019

Вот один из способов их выравнивания.Идея заключается в следующем:

  • Сначала нанесите данные по нижней оси x, используя (ax1)
  • Затем установите предел верхней оси X таким же, какнижней оси x с помощью ax3.set_xlim(ax1.get_xlim())
  • Затем установите отметки верхней оси X в местах, соответствующих нижним значениям оси X (10, 20, 30, ..., 90, 100)
  • Наконец, переименуйте метки галочек, используя ax3.set_xticklabels().

Вот код: я заменяю части, которые уже есть в вашем коде, комментарием #.

# imports and define data and compute accuracy and loss here

# Initiate figure and axis objects here

ax1.set_xlabel('batches')
ax1.set_xticks(np.arange(1, len(batches)+1, 9))
ax1.set_ylabel('accuracy')
ax1.grid()

acc_plt = ax1.plot(batches, accuracy, color='red', label='acc')
loss_plt = ax2.plot(batches, loss, color='blue', label='loss')

ax2.set_ylabel('loss')
ax2.set_yticklabels(np.linspace(3, 10, 9))

new_tick_locations = np.arange(1, 11)*10
new_tick_labels = range(1, 11)

ax3.set_xlabel('epochs')
ax3.set_xlim(ax1.get_xlim())
ax3.set_xticks(new_tick_locations)
ax3.set_xticklabels(new_tick_labels)

# Set legends and show the plot

enter image description here

...