Создайте две метки на одной оси и заполните пробелы выходных - PullRequest
2 голосов
/ 18 марта 2019

Из этого вопроса мне понадобилось несколько советов, чтобы создать гистограмму внизу; Matplotlib - Наложение объема финансов

Но проблема, которую я не могу решить, это пустые места для гистограммы. Как видно из кода, список y3 имеет length=21, и он одинаков для x, y1, y2. Дни, которые являются пустыми, являются выходными, но эти дни удаляются из списка x, прежде чем я создаю какой-либо график. Я не понимаю, как и почему они оставляют пробел, даже после того, как я удалил их из списка моих x-axis.

РЕДАКТИРОВАТЬ: Я сократил вопрос.

]

from datetime import datetime,timedelta
import matplotlib
import matplotlib.pyplot as plt
import matplotlib.dates as mdates

d = datetime.strptime('2019-02-01','%Y-%m-%d')
#create x-axis which consists of all days in february except weekdays, plus the first day in march
x = []
for i in range(0,29):
    if((d + timedelta(days=i)).date().weekday() > 4):
        continue
    x.append((d + timedelta(days=i)).date())

y1 = [317.92, 319.1, 322.73, 324.22, 322.05, 321.06, 323.79, 325.29, 325.55, 327.9, 329.99, 330.92, 331.33, 332.75, 331.41, 333.22, 332.58, 333.64, 331.19, 329.83, 332.46]
y2 = [-0.377507469, -0.3411700881, 0.055641283900000005, 0.44570105590000003, 0.10930618490000005, -0.2948038151, 0.09190098490000004, 0.12858223490000004, 0.09559385990000004, 0.4806987649, 0.11333709890000004, 0.5247366639000001, 0.11605396190000006, -0.30122159309999996, -0.7405398680999999, -0.7053236033999999, -0.33368165239999986, -0.8210733183999999, -1.2753887724, -1.3106836659999999, -1.3450585547999998]
y3 = [27, 15, 12, 4, 5, 11, 4, 14, 18, 19, 13, 17, 28, 22, 4, 15, 26, 21, 21, 21, 9]

fig, ax1 = plt.subplots()
ax1.plot(x, y1, 'b')
ax1.tick_params('y', colors='b')
#create space at the bottom for the bar-plot
pad = 0.25
yl = ax1.get_ylim()
ax1.set_ylim(yl[0]-(yl[1]-yl[0])*pad,yl[1])

fig.suptitle('Some title', fontsize=16)
ax2 = ax1.twinx()

ax2.plot(x, y2,'r')
ax2.tick_params('y', colors='r')
#create space at the bottom for the bar-plot
pad = 0.25
yl = ax2.get_ylim()
ax2.set_ylim(yl[0]-(yl[1]-yl[0])*pad,yl[1])
#create barplot and make it small by adding a (in this case, this is not a modular solution) max-limit
ax3 = ax1.twinx()
ax3.set_ylim(0,250)
ax3.bar(x, y3, color='green', width=1, alpha=0.5)
ax3.set_xlim(min(x),max(x))
ax3.get_yaxis().set_ticks([])

ax1.set_ylabel('1st Value', color='b')
ax2.set_ylabel('2nd Value', color='r')
plt.gcf().autofmt_xdate()
plt.gca().xaxis.set_major_formatter(mdates.DateFormatter('%d-%B'))

plt.show()

1 Ответ

1 голос
/ 19 марта 2019

Ваша ось x не имеет данных для выходных

В следующем коде вы создаете x s, но не включаете выходные.

x = []
for i in range(0,29):
    if((d + timedelta(days=i)).date().weekday() > 4):
        continue
    x.append((d + timedelta(days=i)).date())

Так, например, если 1 февраля - пятница, то первые элементы массива x будут [0, 3, 4, 5].Вы не включаете данные для выходных, но, пропуская [1, 2], вы оставляете место, которое они заняли.

Решение: Удалите места на выходные и отформатируйте метки

Что вы должны сделатьсоздать прокси-данные оси X без пробелов, а затем обработать метки.Полное решение может быть следующим:

from datetime import datetime,timedelta
import matplotlib
import matplotlib.pyplot as plt
import matplotlib.dates as mdates
import matplotlib.ticker as ticker

d = datetime.strptime('2019-02-01','%Y-%m-%d')
#create x-axis which consists of all days in february except weekdays, plus the first day in march
x = []
dates = {}
day = 0
for i in range(0,29):
    if((d + timedelta(days=i)).date().weekday() > 4):
        continue
    x.append(day)
    dates[day] = (d + timedelta(days=i)).date()
    day += 1


def format_date(x, pos=None):
    return dates[int(x)].strftime('%d-%B')

y1 = [317.92, 319.1, 322.73, 324.22, 322.05, 321.06, 323.79, 325.29, 325.55, 327.9, 329.99, 330.92, 331.33, 332.75, 331.41, 333.22, 332.58, 333.64, 331.19, 329.83, 332.46]
y2 = [-0.377507469, -0.3411700881, 0.055641283900000005, 0.44570105590000003, 0.10930618490000005, -0.2948038151, 0.09190098490000004, 0.12858223490000004, 0.09559385990000004, 0.4806987649, 0.11333709890000004, 0.5247366639000001, 0.11605396190000006, -0.30122159309999996, -0.7405398680999999, -0.7053236033999999, -0.33368165239999986, -0.8210733183999999, -1.2753887724, -1.3106836659999999, -1.3450585547999998]
y3 = [27, 15, 12, 4, 5, 11, 4, 14, 18, 19, 13, 17, 28, 22, 4, 15, 26, 21, 21, 21, 9]

fig, ax1 = plt.subplots()
ax1.plot(x, y1, 'b')
ax1.tick_params('y', colors='b')
#create space at the bottom for the bar-plot
pad = 0.25
yl = ax1.get_ylim()
ax1.set_ylim(yl[0]-(yl[1]-yl[0])*pad,yl[1])

fig.suptitle('Some title', fontsize=16)
ax2 = ax1.twinx()

ax2.plot(x, y2,'r')
ax2.tick_params('y', colors='r')
#create space at the bottom for the bar-plot
pad = 0.25
yl = ax2.get_ylim()
ax2.set_ylim(yl[0]-(yl[1]-yl[0])*pad,yl[1])
#create barplot and make it small by adding a (in this case, this is not a modular solution) max-limit
ax3 = ax1.twinx()
ax3.set_ylim(0,250)
ax3.bar(x, y3, color='green', width=1, alpha=0.5)
ax3.set_xlim(min(x),max(x))
ax3.get_yaxis().set_ticks([])

ax1.set_ylabel('1st Value', color='b')
ax2.set_ylabel('2nd Value', color='r')
plt.gcf().autofmt_xdate()
plt.gca().xaxis.set_major_formatter(ticker.FuncFormatter(format_date))

plt.show()

Это создает изображение

Look, no x gaps

, которое представляет собой те же данные, но безпромежутки выходных.

Объяснение различий

Для этого упражнения я изменил три аспекта.Во-первых, я установил x для окончательного представления массива [0, 1, 2, ..., 20] и создал словарь dates для отслеживания того, какой день представляет x[i].

x = []
dates = {}
day = 0
for i in range(0,29):
    if((d + timedelta(days=i)).date().weekday() > 4):
        continue
    x.append(day)
    dates[day] = (d + timedelta(days=i)).date()
    day += 1

Затем я создал функцию format_date длявведите matplotlib.ticker, чтобы обработать метки для оси X.

def format_date(x, pos=None):
    return dates[int(x)].strftime('%d-%B')

Наконец, я установил ось X для использования этой функции формата.При этом используется matplotlib.ticker, который должен быть импортирован.

# in the imports, add
import matplotlib.ticker as ticker

# when describing the x-axis, use
plt.gca().xaxis.set_major_formatter(ticker.FuncFormatter(format_date))
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...