“该策略通过做多基于高管数据计算出的多元化得分最高的五分之一,做空得分最低的五分之一,投资于管理团队多元化的公司。”

I. 策略概要

投资范围包括具有正账面价值和至少12个月过往回报的纽约证券交易所、美国证券交易所和纳斯达克股票。公司的传记数据来自EDGAR数据库的年度报告。多元化是根据高层管理人员的构成来衡量的,如果公司只有一个高层管理人员,则多元化得分为0。投资者做多管理团队最多元化的前五分之一公司,做空管理团队最同质化的后五分之一公司。投资组合每年重新平衡并持有,股票等权重。

II. 策略合理性

作者提出了两种对这种异常现象的解释:质量溢价和错误定价。他们发现,已知的质量因素占该策略阿尔法的比例高达25%。他们还认为,管理层多元化可能是一种尚未被发现的质量因素。错误定价效应与分析师覆盖率有关,分析师关注度越低,股票错误定价的可能性就越高。

III. 来源论文

Diversity Investing [点击查看论文]

<摘要>

高层管理团队的多元化对股票回报至关重要。我们开发了一种新的基于文本的团队多元化衡量标准,并将其应用于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"))

发表评论

了解 Quant Buffet 的更多信息

立即订阅以继续阅读并访问完整档案。

继续阅读