-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathledoit_wolf.py
63 lines (48 loc) · 2 KB
/
ledoit_wolf.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
"""Ledoit & Wolf constant correlation unequal variance shrinkage estimator."""
from typing import Tuple
import numpy as np
def shrinkage(returns: np.array) -> Tuple[np.array, float, float]:
"""Shrinks sample covariance matrix towards constant correlation unequal variance matrix.
Ledoit & Wolf ("Honey, I shrunk the sample covariance matrix", Portfolio Management, 30(2004),
110-119) optimal asymptotic shrinkage between 0 (sample covariance matrix) and 1 (constant
sample average correlation unequal sample variance matrix).
Paper:
http://www.ledoit.net/honey.pdf
Matlab code:
https://www.econ.uzh.ch/dam/jcr:ffffffff-935a-b0d6-ffff-ffffde5e2d4e/covCor.m.zip
Special thanks to Evgeny Pogrebnyak https://github.com/epogrebnyak
:param returns:
t, n - returns of t observations of n shares.
:return:
Covariance matrix, sample average correlation, shrinkage.
"""
t, n = returns.shape
mean_returns = np.mean(returns, axis=0, keepdims=True)
returns -= mean_returns
sample_cov = returns.transpose() @ returns / t
# sample average correlation
var = np.diag(sample_cov).reshape(-1, 1)
sqrt_var = var ** 0.5
unit_cor_var = sqrt_var * sqrt_var.transpose()
average_cor = ((sample_cov / unit_cor_var).sum() - n) / n / (n - 1)
prior = average_cor * unit_cor_var
np.fill_diagonal(prior, var)
# pi-hat
y = returns ** 2
phi_mat = (y.transpose() @ y) / t - sample_cov ** 2
phi = phi_mat.sum()
# rho-hat
theta_mat = ((returns ** 3).transpose() @ returns) / t - var * sample_cov
np.fill_diagonal(theta_mat, 0)
rho = (
np.diag(phi_mat).sum()
+ average_cor * (1 / sqrt_var @ sqrt_var.transpose() * theta_mat).sum()
)
# gamma-hat
gamma = np.linalg.norm(sample_cov - prior, "fro") ** 2
# shrinkage constant
kappa = (phi - rho) / gamma
shrink = max(0, min(1, kappa / t))
# estimator
sigma = shrink * prior + (1 - shrink) * sample_cov
return sigma, average_cor, shrink