В этом посте будет рассмотрено то, как оптимизировать портфель при помощи Python и симуляции Монте Карло. Под оптимизацией портфеля понимается такое соотношение весов, которое будет удовлетворять одному из условий:
- Портфель с минимальным уровнем риском при желаемой доходности;
- Портфель с максимальной доходностью при установленном риске;
- Портфель с максимальным значением доходности
Для расчета возьмем девять акций, которые рекомендовал торговый робот одного из брокеров на начало января 2020 года и так же он устанавливал по ним оптимальные веса в портфеле: 'ATVI','BA','CNP','CMA', 'STZ','GPN','MPC','NEM' и 'PKI'. Для анализа будет взяты данные по акциям за последние три года.
#Загружаем библиотеки
import pandas as pd
import yfinance as yf
import numpy as np
import matplotlib.pyplot as plt
# Получаем данные по акциям
ticker = ['ATVI','BA','CNP','CMA', 'STZ','GPN','MPC','NEM', 'PKI']
stock = yf.download(ticker,'2017-01-01', '2019-01-31')
Если сложить долю всех акций, входящих в портфель, то сумма должна стремиться к единице (а лучше быть равна). Дальше как обычно проведем подготовку данных для расчетов:
# Выделение скорректированой цены закрытия
all_adj_close = stock[['Adj Close']]
# ежедневная доходность
all_returns = all_adj_close.pct_change()
# узнаем среднюю доходность и получаем ковариационную матрицу
mean_returns = all_returns.mean()
cov_matrix = all_returns.cov()
Теперь можно провести расчет для весов предложенных торговым роботом и узнать доходность данного портфеля за последник три года и стандартное отклонение.
#предложение по весам торгового робота
robot = np.array([0.0441, 0.1030, 0.1086, 0.2070, 0.1525, 0.0714, 0.0647, 0.1828, 0.0661])
# доходность, стандартное отклонение и коэффициент Шарпо
portfolio_return_robot = np.sum(mean_returns * robot)
portfolio_std_dev_robot = np.sqrt(np.dot(robot.T,np.dot(cov_matrix, robot)))
sharpo_robot = portfolio_return_robot/portfolio_std_dev_robot
# объединим полученные значения в таблицу и выведем ее
robot_result = np.array([portfolio_return_robot, portfolio_std_dev_robot, sharpo_robot])
robot_result = np.array([portfolio_return_robot, portfolio_std_dev_robot, sharpo_robot])
robot_result = np.concatenate((robot_result, robot), axis=0)
robot_sim_result = pd.DataFrame(robot_result, columns=['Robot'], index=['ret','stdev','sharpe',ticker[0],ticker[1],ticker[2],ticker[3],ticker[4],ticker[5],ticker[6],ticker[7],ticker[8]])
print(robot_sim_result)
Симуляция Монте-Карло
Первоначально небольшое вступительное слово о том, как используется метод Монте-Карла для оптимизации портфеля
Сначала акциям задаются случайные веса, после чего производится расчет доходности и стандартного отклонения. Полученные значения сохраняются. Следующим шагом случайным образом меняются веса (главное не забывать, что их сумма должна составлять единицу) и все повторяется — расчет и сохранение полученного значения. Количество итераций зависит от времени, мощностей компьютера для расчета и рисков, который готов принять инвестор. В этот раз попробуем провести 10000 расчетов для выявления портфеля с минимальным убытком и максимальным значением коэффициента Шарпа.
#создаем массив из нулей
num_iterations = 10000
simulation_res = np.zeros((4+len(ticker)-1,num_iterations))
# сама итерация
for i in range(num_iterations):
#Выбрать случайные веса и нормализовать, чтоб сумма равнялась 1
weights = np.array(np.random.random(9))
weights /= np.sum(weights)
#Вычислить доходность и стандартное отклонение
portfolio_return = np.sum(mean_returns * weights)
portfolio_std_dev = np.sqrt(np.dot(weights.T,np.dot(cov_matrix, weights)))
#Сохранить все полученные значения в массив
simulation_res[0,i] = portfolio_return
simulation_res[1,i] = portfolio_std_dev
#Вычислить коэффициент Шарпа и сохранить
simulation_res[2,i] = simulation_res[0,i] / simulation_res[1,i]
#Сохранить веса
for j in range(len(weights)):
simulation_res[j+3,i] = weights[j]
# сохраняем полученный массив в DataFrame для построения данных и анализа.
sim_frame = pd.DataFrame(simulation_res.T,columns=['ret','stdev','sharpe',ticker[0],ticker[1],ticker[2],ticker[3],ticker[4],ticker[5],ticker[6],ticker[7],ticker[8]])
Теперь можно рассчитать портфель с максимальным коэффициентом Шарпа или минимальным риска.
# узнать максимальный Sharpe Ratio
max_sharpe = sim_frame.iloc[sim_frame['sharpe'].idxmax()]
# узнать минимальное стандартное отклонение
min_std = sim_frame.iloc[sim_frame['stdev'].idxmin()]
print ("The portfolio for max Sharpe Ratio:\n", max_sharpe)
print ("The portfolio for min risk:\n", min_std)
Ну а самое важное представление можно получить, когда данные визуализируешь:
fig, ax = plt.subplots(figsize=(10, 10))
#Создать разноцветный график scatter plot для различных значений коэффициента Шарпо по оси x и стандартного отклонения по оси y
plt.scatter(sim_frame.stdev,sim_frame.ret,c=sim_frame.sharpe,cmap='RdYlBu')
plt.xlabel('Standard Deviation')
plt.ylabel('Returns')
plt.ylim(0,.0015)
plt.xlim(0.007,0.012)
plt.scatter(max_sharpe[1],max_sharpe[0],marker=(5,1,0),color='r',s=600)
plt.scatter(min_std[1],min_std[0],marker=(5,1,0),color='b',s=600)
plt.scatter(portfolio_std_dev_robot, portfolio_return_robot,marker=(5,1,0),color='g',s=600)
plt.show()
Портфель с максимальным коэффициентом Шарпа показан красной звездой, синей — с минимальным стандартным отклонением и зеленой — предложенный роботом. Как видно — портфель, предложенный роботом, не совпадает с этими показателями, но на каком остановиться портфеле — выбор остается за инвестором. А я постараюсь в конце года вернуться к сравнению портфелей. А сейчас все три портфеля находятся в просадке.
sshikov
Вообще-то, методу Монте-Карлы (ну раз вам можно, почему мне нельзя ;) как минимум лет 50. Во всяком случае меня еще в институте в курсе методов оптимизации ему учили. Ну то есть, сам-то метод многократно описан, и реализован.
Поэтому, на мой взгляд, тут стоило бы упомянуть не «как», а «почему». Ответ на этот вопрос для метода оптимизации обычно определяется типом функции — скажем, применять Монте-Карлу для функций гладких как правило нафиг не нужно, так как там хорошо (и быстрее) работают вещи типа градиентного спуска. Так что у нас за функция? Почему именно Монте-Карло?
Zmey56 Автор
Спасибо за развернутое пояснение. Почему его, так мне показался проще для понимания и его я пытался сравнить с роботом. В данный момент я провожу анализы и другими способами, и возможно впоследствии что то напишу, если кто-то более подкованный этого не сделает вперед. Еще раз спасибо за критику.
sshikov
>мне показался проще для понимания
Ну, возможно. Да и в реализации он не сложный. Просто обычно стохастические методы применяют, если функция скажем овражная (или просто не гладкая), ну или что-то в таком духе. Тут вроде не должно быть ничего такого, или мне кажется?
P.S. Поправьте все-таки название — там все еще опечатка (Монте-Карла).
Zmey56 Автор
Спасибо, проглядел что-то в этот раз много орфографических ошибок.
ProLimit
Видимо, потому что в подобных задачах не требуются точные решения, не нужен истинный максимум/минимум, и шаг в любую сторону +- километр не критичен (исходя из довольно сомнительного допущения что прошлые данные как-то влияют на будущие). Зато метод прост в реализации и понятен даже далеким от математики людям.
sshikov
Наверное вы правы (в первую очередь пожалуй насчет того, что допущения в модели очень сильные, и точность нам не особо нужна в такой ситуации), но с другой стороны, стохастические методы могут нам дать совсем не оптимальное решение. И если истинный экстремум все-таки существует (тут мы возвращаемся к исходному вопросу — знаем ли мы, что у нас в итоге за функция? есть ли у нее экстремум? каковы ограничения на решение?), то чуть более другие методы скорее всего будут сильно быстрее. А простота реализации… на мой взгляд все пригодные методы общего назначения давно реализовали.
В принципе, я встречал подходы, когда Монте-Карло используется на первом этапе, чтобы быстро и грубо найти несколько начальных точек для градиентных методов. По-идее, если мы о функции знаем немного, такой подход имел бы право на жизнь.