Momentum investing says that excess returns can be generated by buying recent winners and selling recent losers. In this notebook we will research the momentum factor on our universe of demo stocks. This will help us determine whether we have a profitable idea before turning to a full backtest.
First, load your historical data into pandas.
from quantrocket import get_prices
prices = get_prices("usstock-free-1d", universes="usstock-free", start_date="2018-01-01", end_date="2020-04-01", fields=["Close"])
prices.head()
Sid | FIBBG000B9XRY4 | FIBBG000BFWKC0 | FIBBG000BKZB36 | FIBBG000BMHYD1 | FIBBG000BPH459 | FIBBG000GZQ728 | FIBBG00B3T3HD3 | |
---|---|---|---|---|---|---|---|---|
Field | Date | |||||||
Close | 2018-01-02 | 40.6197 | 116.4411 | 161.6012 | 117.5027 | 80.0809 | 62.6256 | 53.8066 |
2018-01-03 | 40.6126 | 116.9167 | 162.4434 | 118.6251 | 80.4536 | 63.8555 | 53.1531 | |
2018-01-04 | 40.8013 | 117.3845 | 163.7326 | 118.6167 | 81.1617 | 63.9439 | 53.3482 | |
2018-01-05 | 41.2658 | 118.0514 | 165.4429 | 119.5957 | 82.1680 | 63.8923 | 52.7533 | |
2018-01-08 | 41.1125 | 118.5491 | 165.0475 | 119.7476 | 82.2518 | 64.1796 | 53.6408 |
Next, we use closing prices to calculate our momentum factor. We calculate momentum using a twelve-month window but excluding the most recent month, as commonly recommended by academic papers.
closes = prices.loc["Close"]
MOMENTUM_WINDOW = 252 # 12 months = 252 trading days
RANKING_PERIOD_GAP = 22 # 1 month = 22 trading days
earlier_closes = closes.shift(MOMENTUM_WINDOW)
later_closes = closes.shift(RANKING_PERIOD_GAP)
momentum_returns = (later_closes - earlier_closes) / earlier_closes
Now that we have the twelve-month returns, we calculate the next day returns:
next_day_returns = closes.ffill().pct_change().shift(-1)
To see if the twelve-month returns predict next-day returns, we will split the twelve-month returns into bins and look at the mean next-day return of each bin. To do this, we first need to stack our wide-form DataFrames into Series.
momentum_returns = momentum_returns.stack(dropna=False)
next_day_returns = next_day_returns.stack(dropna=False)
Use pandas' qcut
function to create the bins:
import pandas as pd
# For a very small demo universe, you might only want 2 quantiles
num_bins = 2
bins = pd.qcut(momentum_returns, num_bins)
Now group the next day returns by momentum bin and plot the mean return:
next_day_returns.groupby(bins).mean().plot(kind="bar", title="Next-day return by 12-month momentum bin");
For a predictive factor, the higher quantiles should perform better than the lower quantiles.