“该策略交易商品期货,做多表现最佳的合约,做空表现较差的合约,采用风险平价加权方法,并基于12个月的业绩排名和6个月移动平均趋势筛选进行调整。”

I. 策略概要

该策略使用道琼斯-UBS 大宗商品超额回报指数,涵盖 28 种大宗商品,并通过相应的期货进行交易。每月根据过去 12 个月的表现对大宗商品进行四分位排序。投资组合包括表现最好的(赢家)和表现最差的(输家)大宗商品,并采用风险平价方法进行加权,权重与其 60 天波动率成反比。此外,应用趋势跟随过滤器:大宗商品需高于其 6 个月简单移动平均线才能被视为赢家,或低于该均线才能被视为输家。投资者对符合筛选标准的赢家做多,对输家做空,从而构建一个平衡且基于表现的投资组合,同时实现系统性风险管理。

II. 策略合理性

学术研究对趋势跟随策略的历史成功提出了多种解释,包括投资者对新闻的反应不足以及羊群行为。动量效应通常被归因于投资者的非理性行为,因为他们未能完全将新信息纳入交易价格。此外,动量投资者可能利用其他市场参与者的行为偏差(如羊群效应、过度反应、反应不足和确认偏误),以把握可预测的价格趋势并从中获利。

III. 来源论文

Trend Following, Risk Parity and Momentum in Commodity Futures [点击查看论文]

<摘要>

我们证明,将动量策略与趋势跟随策略结合应用于单个商品期货,可以构建出提供有吸引力的风险调整收益的投资组合,其表现优于单纯的动量策略。当我们将这些收益暴露于广泛的系统性风险来源时,发现稳健的阿尔法仍然存在。实验表明,采用风险平价投资组合加权对结果的影响有限,尤其有利于多空策略;相比之下,在风险调整收益和降低下行风险方面,趋势跟随方法的边际影响远远超过动量策略和风险平价调整。总体而言,该策略为商品期货投资提供了一种有吸引力的方法,并强调了趋势跟随策略在商品期货投资中的重要性。

IV. 回测表现

年化回报14.7%
波动率19.33%
β值-0.045
夏普比率0.76
索提诺比率0.089
最大回撤-31.87%
胜率55%

V. 完整的 Python 代码

import numpy as np
from AlgorithmImports import *
class TrendfollowingwithMomentum(QCAlgorithm):
    def Initialize(self):
        self.SetStartDate(1991, 1, 1)
        self.SetCash(100000)
        self.symbols = [
            "CME_S1",   # Soybean Futures, Continuous Contract
            "CME_W1",   # Wheat Futures, Continuous Contract
            "CME_SM1",  # Soybean Meal Futures, Continuous Contract
            "CME_BO1",  # Soybean Oil Futures, Continuous Contract
            "CME_C1",   # Corn Futures, Continuous Contract
            "CME_O1",   # Oats Futures, Continuous Contract
            "CME_LC1",  # Live Cattle Futures, Continuous Contract
            "CME_FC1",  # Feeder Cattle Futures, Continuous Contract
            "CME_LN1",  # Lean Hog Futures, Continuous Contract
            "CME_GC1",  # Gold Futures, Continuous Contract
            "CME_SI1",  # Silver Futures, Continuous Contract
            "CME_PL1",  # Platinum Futures, Continuous Contract
            "CME_CL1",  # Crude Oil Futures, Continuous Contract
            "CME_HG1",  # Copper Futures, Continuous Contract
            "CME_LB1",  # Random Length Lumber Futures, Continuous Contract
            # "CME_NG1",  # Natural Gas (Henry Hub) Physical Futures, Continuous Contract
            "CME_PA1",  # Palladium Futures, Continuous Contract 
            "CME_RR1",  # Rough Rice Futures, Continuous Contract
            "CME_DA1",  # Class III Milk Futures
            "ICE_RS1",  # Canola Futures, Continuous Contract
            "ICE_GO1",  # Gas Oil Futures, Continuous Contract
            "CME_RB2",  # Gasoline Futures, Continuous Contract
            "CME_KW2",  # Wheat Kansas, Continuous Contract
            "ICE_WT1",  # WTI Crude Futures, Continuous Contract
            "ICE_CC1",  # Cocoa Futures, Continuous Contract 
            "ICE_CT1",  # Cotton No. 2 Futures, Continuous Contract
            "ICE_KC1",  # Coffee C Futures, Continuous Contract
            "ICE_O1",   # Heating Oil Futures, Continuous Contract
            "ICE_OJ1",  # Orange Juice Futures, Continuous Contract
            "ICE_SB1",  # Sugar No. 11 Futures, Continuous Contract
        ]
        self.winners = []
        self.losers = []
        self.data = {}
        self.period = 12*21
        self.SetWarmUp(self.period)
     
        for symbol in self.symbols:
            data = self.AddData(QuantpediaFutures, symbol, Resolution.Daily)
            data.SetLeverage(10)
            data.SetFeeModel(CustomFeeModel())
            
            ma = self.SMA(symbol, 6*21, Resolution.Daily)
            self.data[symbol] = SymbolData(symbol, 60, self.period, ma)
        
        self.rebalance_flag: bool = False
        self.Schedule.On(self.DateRules.MonthEnd(self.symbols[0]), self.TimeRules.At(0, 0), self.Rebalance)
    
    def OnData(self, data):
        for symbol, symbol_data in self.data.items():
            if self.securities[symbol].get_last_data() and self.time.date() > QuantpediaFutures.get_last_update_date()[symbol]:
                self.liquidate(symbol)
                symbol_data.History.reset()
                continue
            symbol_obj = self.Symbol(symbol)
            if symbol_obj in data.Keys:
                if data[symbol_obj]:
                    price = data[symbol_obj].Value
                    if price != 0:
                        self.data[symbol].Update(price)
        
        if not self.rebalance_flag:
            return
        self.rebalance_flag = False
        targets: List[PortfolioTarget] = []
        for i, portfolio in enumerate([self.winners, self.losers]):
            for symbol_data in portfolio:
                if symbol_data[1].Weight != 0:
                    if symbol_data[0] in data and data[symbol_data[0]]:
                        targets.append(PortfolioTarget(symbol_data[0], ((-1) ** i) * symbol_data[1].Weight))
        self.SetHoldings(targets, True)
        self.winners.clear()
        self.losers.clear()
    def Rebalance(self):
        # Return sorting
        return_values = []
        for data in self.data.items():
            if data[1].IsReady():
                return_values.append(data[1].Return())
        
        if len(return_values) == 0: return
    
        high_percentile = np.percentile(return_values, 75)
        low_percentile = np.percentile(return_values, 25)
        winners_by_ret = list(data for data in self.data.items() if data[1].IsReady() and data[1].Return() > high_percentile)
        losers_by_ret = list(data for data in self.data.items() if data[1].IsReady() and data[1].Return() < low_percentile)
        
        # Weighting
        total_vol = sum((1.0/data[1].Volatility()) for data in winners_by_ret if data[1].IsReady()) + sum((1.0/data[1].Volatility()) for data in losers_by_ret if data[1].IsReady())
        for data in winners_by_ret + losers_by_ret:
            if data[1].IsReady():
                vol = data[1].Volatility()
                data[1].Weight = (1.0 / vol) / total_vol
        # Trend sorting
        self.winners = list(data for data in winners_by_ret if data[1].Price > data[1].MA.Current.Value)
        self.losers = list(data for data in losers_by_ret if data[1].Price < data[1].MA.Current.Value)
        self.rebalance_flag = True
class SymbolData:
    def __init__(self, symbol, volatility_lookback, return_lookback, ma):
        self.Symbol = symbol
        self.History = RollingWindow[float](return_lookback)
        self.Price = 0.0
        self.MA = ma
        self.Weight = 0.0
        self.Volatility_lookback = volatility_lookback
    def IsReady(self) -> bool:
        return self.History.IsReady
            
    def Update(self, value: float):
        self.Price = value
        self.History.Add(float(value))
    
    def Return(self) -> float:
        prices = [x for x in self.History]
        return prices[0] / prices[-1] - 1
    
    def Volatility(self) -> float:
        prices = np.array([x for x in self.History])[-self.Volatility_lookback:]
        returns = prices[:-1] / prices[1:] - 1
        return np.std(returns) * np.sqrt(252)
    
# Quantpedia data.
# NOTE: IMPORTANT: Data order must be ascending (datewise)
class QuantpediaFutures(PythonData):
    _last_update_date:Dict[Symbol, datetime.date] = {}
    @staticmethod
    def get_last_update_date() -> Dict[Symbol, datetime.date]:
       return QuantpediaFutures._last_update_date
    def GetSource(self, config, date, isLiveMode):
        return SubscriptionDataSource("data.quantpedia.com/backtesting_data/futures/{0}.csv".format(config.Symbol.Value), SubscriptionTransportMedium.RemoteFile, FileFormat.Csv)
    def Reader(self, config, line, date, isLiveMode):
        data = QuantpediaFutures()
        data.Symbol = config.Symbol
        
        if not line[0].isdigit(): return None
        split = line.split(';')
        
        data.Time = datetime.strptime(split[0], "%d.%m.%Y") + timedelta(days=1)
        data['back_adjusted'] = float(split[1])
        data['spliced'] = float(split[2])
        data.Value = float(split[1])
        if config.Symbol.Value not in QuantpediaFutures._last_update_date:
            QuantpediaFutures._last_update_date[config.Symbol.Value] = datetime(1,1,1).date()
        if data.Time.date() > QuantpediaFutures._last_update_date[config.Symbol.Value]:
            QuantpediaFutures._last_update_date[config.Symbol.Value] = data.Time.date()
        return data
# Custom fee model.
class CustomFeeModel(FeeModel):
    def GetOrderFee(self, parameters):
        fee = parameters.Security.Price * parameters.Order.AbsoluteQuantity * 0.00005
        return OrderFee(CashAmount(fee, "USD"))

发表评论

了解 Quant Buffet 的更多信息

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

继续阅读