
“该策略通过做多基于高管数据计算出的多元化得分最高的五分之一,做空得分最低的五分之一,投资于管理团队多元化的公司。”
资产类别: 股票 | 地区: 美国 | 周期: 每年 | 市场: 股票 | 关键词: 管理层、多元化
I. 策略概要
投资范围包括具有正账面价值和至少12个月过往回报的纽约证券交易所、美国证券交易所和纳斯达克股票。公司的传记数据来自EDGAR数据库的年度报告。多元化是根据高层管理人员的构成来衡量的,如果公司只有一个高层管理人员,则多元化得分为0。投资者做多管理团队最多元化的前五分之一公司,做空管理团队最同质化的后五分之一公司。投资组合每年重新平衡并持有,股票等权重。
II. 策略合理性
作者提出了两种对这种异常现象的解释:质量溢价和错误定价。他们发现,已知的质量因素占该策略阿尔法的比例高达25%。他们还认为,管理层多元化可能是一种尚未被发现的质量因素。错误定价效应与分析师覆盖率有关,分析师关注度越低,股票错误定价的可能性就越高。
III. 来源论文
Diversity Investing [点击查看论文]
- 阿尔贝托·曼科尼,A. 埃马努埃莱·里佐 和 奥利弗·斯帕尔特。博科尼大学 – 金融系;经济政策研究中心 (CEPR)。新里斯本商学院。曼海姆大学 – 商学院;欧洲公司治理研究院 (ECGI)。
<摘要>
高层管理团队的多元化对股票回报至关重要。我们开发了一种新的基于文本的团队多元化衡量标准,并将其应用于2001年至2014年美国公司的40,000多名高层管理人员的样本。在价值加权的基础上,购买团队多元化的公司并出售团队同质化的公司——我们称之为“多元化投资”的策略——在我们样本期内优于主要的资产定价异常现象。我们研究了一系列可能的解释,并找到了强有力的证据,表明分析师和投资者对团队多元化的公司的回报预期存在向下偏差,这与多元化回报的错误定价解释一致。


IV. 回测表现
| 年化回报 | 6.55% |
| 波动率 | 7.38% |
| β值 | 0.769 |
| 夏普比率 | 0.89 |
| 索提诺比率 | 0.236 |
| 最大回撤 | N/A |
| 胜率 | 77% |
V. 完整的 Python 代码
from AlgorithmImports import *
from io import StringIO
from pandas.core.frame import dataframe
from typing import List, Dict
import pandas as pd
# endregion
class ManagementDiversityStrategy(QCAlgorithm):
def Initialize(self):
self.SetStartDate(2000, 1, 1)
self.SetCash(100000)
self.market: Symbol = self.AddEquity('SPY', Resolution.Daily).Symbol
self.leverage: int = 3
self.selection_month: int = 6
self.short_allocation: float = -.0
self.weights: Dict[Symbol, float] = {}
# source: https://www.fair360.com/top-50-list/2023/
top_diversity_firms: str = self.Download('data.quantpedia.com/backtesting_data/economic/top50_diversity_firms.csv')
self.top_diversity_firms_df: dataframe = pd.read_csv(StringIO(top_diversity_firms), delimiter=';')
self.selection_flag: bool = False
self.settings.daily_precise_end_time = False
self.Settings.MinimumOrderMarginPortfolioPercentage = 0.
self.UniverseSettings.Resolution = Resolution.Daily
self.AddUniverse(self.FundamentalSelectionFunction)
self.Schedule.On(self.DateRules.MonthStart(self.market), self.TimeRules.AfterMarketOpen(self.market), self.Selection)
def OnSecuritiesChanged(self, changes: SecurityChanges) -> None:
for security in changes.AddedSecurities:
security.SetFeeModel(CustomFeeModel())
security.SetLeverage(self.leverage)
def FundamentalSelectionFunction(self, fundamental: List[Fundamental]) -> List[Symbol]:
# selection in the beginning of June
if not self.selection_flag:
return Universe.Unchanged
selected: Dict[str, Fundamental] = {
x.Symbol.Value: x for x in fundamental if x.HasFundamentalData and x.MarketCap != 0
}
long: List[Fundamental] = []
if str(self.Time.year) in list(self.top_diversity_firms_df.columns):
long = [selected[x] for x in self.top_diversity_firms_df[str(self.Time.year)].values if x in selected]
else:
self.Liquidate()
if len(long) != 0:
# calculate weights based on values
self.weights[self.market] = self.short_allocation
sum_long: float = sum([x.MarketCap for x in long])
for stock in long:
self.weights[stock.Symbol] = stock.MarketCap / sum_long
return list(self.weights.keys())
def OnData(self, data: Slice) -> None:
# yearly rebalance
if not self.selection_flag:
return
self.selection_flag = False
targets: List[PortfolioTarget] = [PortfolioTarget(symbol, weight) for symbol, weight in self.weights.items() if symbol in data and data[symbol]]
self.SetHoldings(targets, True)
self.weights.clear()
def Selection(self) -> None:
if self.Time.month == self.selection_month:
self.selection_flag = True
# Custom fee model
class CustomFeeModel(FeeModel):
def GetOrderFee(self, parameters: OrderFeeParameters) -> OrderFee:
fee: float = parameters.Security.Price * parameters.Order.AbsoluteQuantity * 0.00005
return OrderFee(CashAmount(fee, "USD"))