It is worth reiterating:
Nothing I am about to say should be taken as advice for investing. These results are based on prior observed returns and the future rarely mimics the past. These techniques can give helpful insight on how you can better allocate a portfolio. It should not be used as the sole investment decision. Speak with a qualified professional if you are looking for advice.
Building on the work of Markowitz, Treynor, Sharpe, et. al. developed the Capital Asset Pricing Model -- CAPM. They shared the Nobel Prize with Markowitz in 1990 for this work. CAPM is a general equilibrium model. The model assumes that market prices reflect all available information and give the "fair" value of a security. Based on that, the market portfolio can be shown to be market capitalization weighted portfolio. Market capitalization (or market cap) is defined as the share price times the number of shares outstanding -- the total value of the equity of a company. A company's weight is its market cap divided by the total market cap of all securities.
Capitalization weighting has become the standard for indexes and index funds. The S&P500 being what most people consider the standard "market portfolio." We will test our portfolio optimization strategy against a capitalization weighted strategy.
Now there are problems with CAPM. There are lots of ways to poke holes in it. One way is to say that prices are not exactly fair value, but instead mean revert to fair value. In this case, when a price is above fair value, the market cap weighted portfolio will overweight the overpriced security. When it mean reverts, the cap weight portfolio will under perform because of the overweighting.
This position was famously put forward by Robert Arnott. I highly recommend his book,The Fundamental Index: A Better Way to Invest. He argues that any portfolio strategy that breaks the correlation with price will outperform the capitalization index over time. He created a new index which he puts forward in the book. Another would be to simply equal weight each security (S&P actually publishes this index). Because of that, we will also test our strategy against an equal weight strategy.
Here is our portfolio optimization strategy:
- At the beginning of each quarter, take the previous quarterly returns and calculate the market portfolio.
- Use this portfolio for the current quarter.
- At the start of the next quarter, go back to #1.
- Require at least 3 stocks in our portfolio.
- No shorting.
- Use 2% as the risk free rate.
- For the first quarter and if the optimization fails, use an equal weight portfolio.
This strategy will tend to outperform when prices are trending quarter over quarter. I.E. if last quarter's returns and volatility accurately predict this quarter's values. We are not taking trading costs into account. The 2% rate is static, and we should technically use the 3 month treasury rate for each quarter beginning. That's why this is just an illustration.
To start, we need to modify the marketPortfolio() function we previously created. You can find it here. The new function signature is
marketPortfolio = function(merged, rf, returnNames, weightNames, graph=FALSE, points=500, maxWeight=.334, Debug=FALSE)
The improvements we have made are:
- Add a Debug option to print outputs if we request them
- Make the function fault tolerant. Sometimes a feasible solution is not possible and the function needs to detect this and handle it accordingly.
- Cap the max return we search over to 50%. This is a sanity check as large values can cause other weird behavior.
- Likewise, put a floor on the min return at .005%.
- If the max return is <0, then simply find the minimum variance portfolio that achieves that value.
- Add a maxWeight option that lets us cap the weight on any 1 security.
The universe of stocks we will consider is 28 of the Dow components (for some reason Yahoo! Finance only gave me 28 instead of 30). I pre-downloaded the returns and stored them in a .rda file. You can get it here. I calculated the starting market cap weights of each stock and stored them in a .csv file. Get it here.
#read the saved returns
load("D:\\Workspace\\PortfolioOptimization\\returns.rda")
returns = results
rm(results)
stocks = colnames(returns)
#get the capitalization weights
stockWgt = read.table("D:\\Workspace\\PortfolioOptimization\\stocks.csv",
header=TRUE,sep=",")[,"Weight"]
#calculate the CapWeight
portfolio returns
results = as.matrix(returns) %*% stockWgt
colnames(results) = c("CapWeight
Portfolio")
results = as.timeSeries(results)
#calculate the EqualWeight
portfolio Returns
ret = t(as.matrix(rep(1/length(stocks),length(stocks))))
resEqual = as.matrix(returns) %*% t(ret)
colnames(resEqual) = "EqualWeight
Portfolio"
#combing the results into 1
timeSeries object
results = cbind(results,resEqual)
#grab the dates of the returns
dates = time(returns)
dates = dates@Data
#get the Quarters for the dates
qtrs = quarters(dates)
qtrs = cbind(dates,qtrs)
keep = NULL
lastQtr = "n"
#go through the dates and
quaters and keep only the date with
#the first day of the quarter
for (i in seq(1,nrow(qtrs))){
if (qtrs[i,2] == lastQtr){
if (i == nrow(qtrs)){
keep
= c(keep,1)
}
else {
keep
= c(keep,0)
}
}else {
keep
= c(keep,1)
}
lastQtr
= qtrs[i,2]
}
qtrs = cbind(qtrs,keep)
#get the indices of the first
days of the quarter
indx = which(qtrs[,3] == 1)
#for the first quarter, use the
equal weights
res = as.matrix(returns[indx[1]:(indx[2]-1),]) %*% t(ret)
#loop throug the quarters,
calculate the market portfolio
#based on the previous
quarter's data and then calculate
#the returns for the current
quarter
for (i in seq(2,length(indx)-1)){
print("Running
")
print(i)
#get subset of
last quarter stock returns
subset
= returns[indx[i-1]:(indx[i]-1),]
s
= start(subset)
e
= end(subset)
print("Fitting
for:")
print(s)
print(e)
#calculate
market portfolio
mp
= marketPortfolio(subset,.02,stocks,stocks,
graph=TRUE,
points=500,
Debug=FALSE)
#if
optimization failed, use equal weights, else get
#the weights
from the market portfolio
if (is.null(mp)){
ret
= t(as.matrix(rep(1/length(stocks),length(stocks))))
}
else {
ret
= as.matrix(mp[,stocks])
}
#subset for
the current quarter
subRes
= returns[indx[i]:(indx[i+1]-1),]
s
= start(subRes)
e
= end(subRes)
print("Calculating
Returns for:")
print(s)
print(e)
#calculate
current quarter returns for market
#portfolio and
add to return series
subRes
= as.matrix(subRes) %*% t(ret)
res
= rbind(res,subRes)
}
#loop does not calculate return
for the last day in the series
subRes = returns[nrow(returns),]
subRes = as.matrix(subRes) %*% t(ret)
res = rbind(res,subRes)
#add the porfoliot optimized
strategy to the others
colnames(res) = "Portfolio
Optimization"
res = as.timeSeries(res)
results = cbind(results,res)
#calculate annualized return
statistics
table.AnnualizedReturns(results)
#calculate and chart the
correlations
png("d:\\Workspace\\PortfolioOptimization\\mpCorr.png")
chart.Correlation(results,histogram=TRUE,pch="+")
dev.off();
#calculate and chart the
cumlative returns
png("d:\\Workspace\\PortfolioOptimization\\mpReturns.png")
chart.CumReturns(results,
main="Total
Returns CapWeight vs PortOpt",
legend.loc="topleft")
dev.off()
Our Annualized Return Table:
CapWeight
Portfolio
|
EqualWeight
Portfolio
|
Portfolio
Optimization
|
|
Annualized
Return
|
-0.0393
|
0.0128
|
0.0069
|
Annualized
Std Dev
|
0.2530
|
0.2242
|
0.1785
|
Annualized
Sharpe (Rf=0%)
|
-0.1554
|
0.0570
|
0.0387
|
Our portfolio optimization portfolio outperforms the Cap Weight, but under performs the Equal Weight. Not surprising if you believe Mr. Arnott as we did not break the correlation with price.
Here is the chart of correlations:
We've created a portfolio that is correlated with the "market" cap weight portfolio, but not nearly so as the equal weight portfolio. The degree of correlation of the equal weight and the cap weight is interesting.
Here is the time series of returns charted:
Interestingly, our portfolio in green has trended sharply up since the market bottom in March of 2009. It strongly outperformed the Cap Weight portfolio.
No comments:
Post a Comment