Momentum trading refers to a general class of strategies which extrapolate from existing trends. They assume that stocks which are going up will continue to go up and stocks which are going down will continue going down, and buy and sell accordingly. This is in contrast to mean-reversion strategies, which rely on trends reversing direction.

For instance, below we see the price of an asset fluctuating but overall going up.

from statsmodels import regression

import statsmodels.api as sm

import scipy.stats as stats

import scipy.spatial.distance as distance

import matplotlib.pyplot as plt

import pandas as pd

import numpy as np

# Get the price of the stock
start = '2014-01-01'
end = '2015-01-01'
asset = get_price('600196.XSHG', fields='price', start_date=start, end_date=end)
dates = asset.index
# Draw an image of price over time
_, ax = plt.subplots()
ax.plot(asset)
ticks = ax.get_xticks()
ax.set_xticklabels([dates[i].date() for i in ticks[:-1]]) # Label x-axis with dates
# Fitting
X = np.arange(len(asset))
x = sm.add_constant(X)
model = regression.linear_model.OLS(asset, x).fit()
a = model.params[0]
b = model.params[1]
Y_hat = X * b + a
plt.plot(X, Y_hat, 'r', alpha=0.9);
plt.ylabel('Price')
plt.legend(['600196.XSHG', 'Trendline']);

The hypothesis here is that the motion of the stock is based on some upwards trend, captured by the linear model, plus some random noise.
# Get price data for the asset
# Load pricing data for an asset
start = '2015-01-01'
end = '2015-06-01'
asset = get_price('600196.XSHG', fields='price', start_date=start, end_date=end)
dates = asset.index
# Draw a curve of asset prices over time
_, ax = plt.subplots()
ax.plot(asset)
ticks = ax.get_xticks()
ax.set_xticklabels([dates[i].date() for i in ticks[:-1]]) # Label x-axis with dates
# Fitting
X = np.arange(len(asset))
x = sm.add_constant(X) # Add a column 1 to the Independent variable to find the intercept
model = regression.linear_model.OLS(asset, x).fit()
a = model.params[0] # Find the intercept of the fitted curve
b = model.params[1]
Y_hat = X * b + a
plt.plot(X, Y_hat, 'r', alpha=0.9);
plt.ylabel('Price')
plt.legend(['60096.XSHG', 'Trendline']);
Momentum and Auto-Correlation
Many things in finance are autocorrelated. An intuitive explanation for this is that prices are determined by traders, which take into account past prices when determining new ones. The formula might be
Pt=Tt+ϵ
Tt=X+P1,t−1
Pt=X+P1,t−1+ϵ
Where X covers all the other factors traders consider and  ϵ  is random noise. This is the definition of an autocorrelated series, in which the value at time  t  depends on previous values.
Some forms of autocorrelation will not exhibit momentum behavior. For instance, a very simple autocorrelation relation could be as follows.
def generate_autocorrelated_values(N):
X = np.zeros(N)
for i in range(N-1):
X[i+1] = X[i] + np.random.normal(0, 1)
return X
for i in range(10):
X = generate_autocorrelated_values(100)
plt.plot(X)
plt.xlabel('$t$')
plt.ylabel('$X_t$');

These series will not exhibit momentum behavior, as the amount they go up and down each timestep is purely random. However, if the series’ returns are autocorrelated, then we will see momentum behavior. We can simulate this as follows.

def generate_autocorrelated_values(N):
X = np.zeros(N)
for i in range(1, N-1):
# Find historical income
past_returns = X[i] - X[i-1]
# Assume that the future rate of return is the same as the current rate of return at that price
X[i+1] = X[i] + past_returns + np.random.normal(0, 1)
return X
for i in range(10):
X = generate_autocorrelated_values(10)
plt.plot(X)
plt.xlabel('$t$')
plt.ylabel('$X_t$');
It's now clear to see that if investors incorporate information about past returns into the future price, we will see momentum behavior. Obviously in real data this will be far more noisy and difficult to observe.
Testing for Autocorrelation
We can test for autocorrelation in a few ways. One is by checking for a unit root. If the series has a unit root, we should be worried that there is autocorrelation present.
from statsmodels.tsa.stattools import adfuller
X1 = generate_autocorrelated_values(100)
X2 = np.random.normal(0, 1, 100)
# Calculate the p-value of the Dickey-Fuller test to test the null hypothesis: this sequence has a unit root
print 'X1'
_, pvalue, _, _, _, _ = adfuller(X1)
if pvalue > 0.05:
print ' Cannot reject the null hypothesis that this sequence has a unit root'
else:
print ' Reject the null hypothesis that this sequence has a unit root'
print 'X2'
_, pvalue, _, _, _, _ = adfuller(X2)
if pvalue > 0.05:
print ' Cannot reject the null hypothesis that this sequence has a unit root'
else:
print ' Reject the null hypothesis that this sequence has a unit root'
Momentum vs. Mean Reversion
Momentum and mean reversion are in a sense opposite strategies. In a momentum model one hypothesizes that past upwards movement is indicative of future upwards movement. In a mean reversion model one hypothesizes that past upward trends are simply temporary overvaluings and the price will return to normal. Both are important models that work in different cases.
It is important to check for which kind of behavior might be present in your data before you actually try developing a strategy based on that.
Time Frame is Important
Momentum behavior and mean reversion behavior can both occur in the same asset. Take for example XLY again. We can see a trend upwards, but there is also movement around that trend. One approach could be to buy an hold XLY and attempt to make money on the longer timeframe. Another approach would be to buy XLY whenever it went under the trendline and sell it whenever it was beneath. This strategy would work on shorter timeframes.
# Get price data
start = '2014-01-01'
end = '2015-01-01'
asset = get_price('600196.XSHG', fields='price', start_date=start, end_date=end)
dates = asset.index
# Draw an image of price over time
_, ax = plt.subplots()
ax.plot(asset)
ticks = ax.get_xticks()
ax.set_xticklabels([dates[i].date() for i in ticks[:-1]]) # Label the x-axis with a date
# Fitting
X = np.arange(len(asset))
x = sm.add_constant(X) # Add a constant column to the Independent variable to calculate the intercept
model = regression.linear_model.OLS(asset, x).fit()
a = model.params[0] # Calculating the intercept
b = model.params[1]
Y_hat = X * b + a
plt.plot(X, Y_hat, 'r', alpha=0.9);
plt.ylabel('Price')
plt.legend(['600196', 'Trendline']);
To see how a mean reversion signal might look, let's look at the difference between the asset and the prediction. Sure enough, looks like some good opportunity for mean reversion as the price difference crosses the zero line many times.
Time Frame is Important for Predictive Capacity
Each momentum measure will be predictive over different time horizons. Some over years and some over days. This will also be true of different asset classes. Some asset classes have momentum over years and some over days. It is important, when constructing a momentum model, to keep in mind what time frame makes sense. That time frame should be roughly the time frame on which you rebalance/exit the trade. It should also be the time frame you use when measuring predictive capacity of your model.
IMPORTANT: Model Selection
It is important not to fall prey to statistical biases by throwing a lot of models at a dataset and seeing which fit. To understand why this is true, see other lectures in the series, especially The Dangers of Overfitting. In short, if you throw a lot of models at a dataset, one will likely happen to look good based on random chance. This model will not have predictive power on any new data, just the data you've already seen.
A better approach is to let your model creation be driven by underlying things you know to be true about your data. For instance, here is a good development path:
Hypothesize that based on investor psychology, certain stocks should exhibit momentum behavior.
Construct a mathemtical model that follows this hypothesis.
Test your hypothesis by seeing if that model fits your data well.
Test your model out of sample on new data you've never seen before.
Trading on Momentum

Measuring Momentum

To trade on momentum, we must first be able to measure it. For information on how to measure momentum, please see the Measuring Momentum lecture. Various ways to quantify momentum will be discussed there. We will discuss what you might do once you have developed a measure of momentum here.

All following examples will assume that a good measure of momentum p, has been developed. The notation p(At) will denote the momentum of asset A and time t.

Entry and Exit Signals

One way to use momentum is to have cutoffs and enter a position when momentum is significantly high, and leave it when momentum is significantly low. The can be done for shorting stocks as well if momentum is in the negative direction. Good examples of this would be moving average crossover strategies, as discussed in the Measuing Momentum lecture.

Ranking Stocks for Long-Short Baskets

Another approach is to use momentum as a factor for ranking stocks. The general approach and its benefits are descrbed in the Long-Short Equity and Ranking Universes by Factors lectures. In short, you make a bet that stocks with higher momentum will outperform those with lower momentum. This tends to be more market neutral and robust. Momentum could also be used as one of many factors in a ranking scheme.

Cross-Sectional Momentum

Cross sectional momentum is when rather than looking at absolute momentum, you look at momentum of every stock relative to every other stock. This has the advantage of providing a relative scale, rather than a somewhat meaningless absolute measure of momentum. An example of this can be found in the algorithm associated with this lecture. To access it, go to the lectures page, hover over this lecture, and click “Get Backtest.”

Leave a Reply

Your email address will not be published. Required fields are marked *