Stratégie de réversion de la moyenne mobile double

Auteur:ChaoZhang est là., Date: 2023-11-17 16h56 et 24h
Les étiquettes:

img

Résumé

La stratégie de réversion des moyennes mobiles doubles est une stratégie de trading de réversion moyenne à court terme typique.

La logique de la stratégie

La stratégie utilise deux moyennes mobiles pour générer des signaux de trading. La première MA maopening est utilisée pour déterminer la direction de la tendance. La deuxième MA maclosing est utilisée pour générer des signaux de trading.

Lorsque le maopening monte, il indique que le marché actuel est en tendance haussière. Lorsque le maopening descend, il indique que le marché actuel est en tendance baissière. le maclosing est multiplié par un coefficient supérieur à 1 pour le rendre plus sensible à la génération de signaux d'inversion précoces.

Plus précisément, lorsque l'ouverture de ma va vers le haut et que la fermeture de ma traverse le bas de ma ouverture, cela indique un renversement de tendance. La stratégie ouvrira une position courte. Lorsque l'ouverture de ma va vers le bas et que la fermeture de ma traverse le haut de ma ouverture, cela indique un renversement de tendance. La stratégie ouvrira une position longue.

Les paramètres de la stratégie comprennent le type de MA, la longueur, la source de données, etc. Les performances de trading peuvent être optimisées en ajustant ces paramètres.

Analyse des avantages

Les principaux avantages de la stratégie de réversion à double MA sont les suivants:

  1. Les moyennes mobiles rapides peuvent rapidement capturer les inversions à court terme avec des retraits plus petits.

  2. Le croisement de deux MAs génère des signaux de trading clairs.

  3. Très configurable avec plusieurs paramètres réglables. Les paramètres de deux MA et coefficients peuvent être optimisés.

  4. Facile à automatiser avec un flux logique clair, la logique simple et le trading haute fréquence le rendent très adapté au trading automatisé.

  5. Risque contrôlable avec mécanisme de stop loss.

Analyse des risques

Il y a aussi des risques liés à cette stratégie:

  1. Le décalage des signaux de croisement MA. Les MA eux-mêmes sont en retard par rapport au prix. Le croisement peut se produire après que la tendance se soit inversée pendant un certain temps.

  2. Le risque d'une transaction à la pioche: la tendance inversée peut rapidement s'inverser, entraînant des pertes consécutives.

  3. Bien que le stop loss limite la perte unique, un stop loss consécutif peut encore entraîner de gros drawdowns.

  4. Risque de surajustement: une optimisation excessive des paramètres peut entraîner un surajustement et de mauvaises performances dans le trading en direct.

Les solutions sont les suivantes:

  1. Optimisez les paramètres pour trouver des MAs plus rapides.

  2. Ajoutez des filtres, tels que des indicateurs de volume et de volatilité, pour éviter les transactions à la scie.

  3. Ajuster la position de stop loss pour réduire la probabilité d'un stop loss consécutif.

  4. Test de robustesse des ensembles de paramètres pour évaluer les risques de surmonture.

Directions d'amélioration

La stratégie peut être encore optimisée dans les aspects suivants:

  1. Testez différents types d'AM pour en trouver de plus sensibles, comme le KAMA, le ZLEMA, etc.

  2. Optimiser les longueurs de MA pour trouver la combinaison optimale.

  3. Testez différentes sources de données, telles que le prix de clôture, le prix médian, le prix typique, etc.

  4. Ajoutez un filtre de tendance pour éviter les signaux de renversement inappropriés, comme le canal de Donchian.

  5. Ajouter d'autres indicateurs pour la confirmation, tels que MACD, OBV, etc.

  6. Améliorer les mécanismes de gestion des risques, tels que le déplacement du stop loss, la perte maximale du compte, etc.

  7. Optimisation du portefeuille pour trouver l'allocation optimale des actifs.

  8. Test de robustesse des paramètres pour évaluer les risques de surmonture.

Conclusion

Le double MA est une stratégie de trading à court terme simple et pratique. Il est adapté pour capturer les retours à court terme avec le trading quantitatif. Cependant, des risques tels que les retards et les transactions à ciseaux existent. La stratégie peut être améliorée en optimisant les paramètres, en ajoutant des filtres, en améliorant le contrôle des risques, etc. pour développer une stratégie stable et efficace avec de bonnes performances de trading réelles.


/*backtest
start: 2023-10-17 00:00:00
end: 2023-11-16 00:00:00
period: 1h
basePeriod: 15m
exchanges: [{"eid":"Futures_Binance","currency":"BTC_USDT"}]
*/

//@version=5
strategy(title = "hamster-bot MRS 2", overlay = true, default_qty_type = strategy.percent_of_equity, initial_capital = 100, default_qty_value = 100, pyramiding = 9, commission_value = 0.045, backtest_fill_limits_assumption = 1)
info_options = "Options"

on_close = input(false, title = "Entry on close", inline=info_options, group=info_options)
OFFS = input.int(0, minval = 0, maxval = 1, title = "| Offset View", inline=info_options, group=info_options)
trade_offset = input.int(0, minval = 0, maxval = 1, title = "Trade", inline=info_options, group=info_options)
use_kalman_filter = input.bool(false, title="Use Kalman filter", group=info_options)

//MA Opening
info_opening = "MA Opening"
maopeningtyp = input.string("SMA", title="Type", options=["SMA", "EMA", "TEMA", "DEMA", "ZLEMA", "WMA", "Hma", "Thma", "Ehma", "H", "L", "DMA"], title = "", inline=info_opening, group=info_opening)
maopeningsrc = input.source(ohlc4, title = "", inline=info_opening, group=info_opening)
maopeninglen = input.int(3, minval = 1, maxval = 200, title = "", inline=info_opening, group=info_opening)

//MA Closing
info_closing = "MA Closing"
maclosingtyp = input.string("SMA", title="Type", options=["SMA", "EMA", "TEMA", "DEMA", "ZLEMA", "WMA", "Hma", "Thma", "Ehma", "H", "L", "DMA"], title = "", inline=info_closing, group=info_closing)
maclosingsrc = input.source(ohlc4, title = "", inline=info_closing, group=info_closing)
maclosinglen = input.int(3, minval = 1, maxval = 200, title = "", inline=info_closing, group=info_closing)
maclosingmul = input.float(1, step = 0.005, title = "mul", inline=info_closing, group=info_closing)

long1on    = input(true, title = "", inline = "long1")
long1shift = input.float(0.96, step = 0.005, title = "Long", inline = "long1")
long1lot   = input.int(10, minval = 0, maxval = 10000, step = 10, title = "Lot 1", inline = "long1")
short1on    = input(true, title = "", inline = "short1")
short1shift = input.float(1.04, step = 0.005, title = "short", inline = "short1")
short1lot   = input.int(10, minval = 0, maxval = 10000, step = 10, title = "Lot 1", inline = "short1")
startTime = input(timestamp("01 Jan 2010 00:00 +0000"), "Start date", inline = "period")
finalTime = input(timestamp("31 Dec 2030 23:59 +0000"), "Final date", inline = "period")

HMA(_src, _length) =>  ta.wma(2 * ta.wma(_src, _length / 2) - ta.wma(_src, _length), math.round(math.sqrt(_length)))
EHMA(_src, _length) =>  ta.ema(2 * ta.ema(_src, _length / 2) - ta.ema(_src, _length), math.round(math.sqrt(_length)))
THMA(_src, _length) =>  ta.wma(ta.wma(_src,_length / 3) * 3 - ta.wma(_src, _length / 2) - ta.wma(_src, _length), _length)
tema(sec, length)=>
    tema1= ta.ema(sec, length)
    tema2= ta.ema(tema1, length)
    tema3= ta.ema(tema2, length)
    tema_r = 3*tema1-3*tema2+tema3
donchian(len) => math.avg(ta.lowest(len), ta.highest(len))
ATR_func(_src, _len)=>
    atrLow = low - ta.atr(_len)
    trailAtrLow = atrLow
    trailAtrLow := na(trailAtrLow[1]) ? trailAtrLow : atrLow >= trailAtrLow[1] ? atrLow : trailAtrLow[1]
    supportHit = _src <= trailAtrLow
    trailAtrLow := supportHit ? atrLow : trailAtrLow
    trailAtrLow
f_dema(src, len)=>
    EMA1 = ta.ema(src, len)
    EMA2 = ta.ema(EMA1, len)
    DEMA = (2*EMA1)-EMA2
f_zlema(src, period) =>
    lag = math.round((period - 1) / 2)
    ema_data = src + (src - src[lag])
    zl= ta.ema(ema_data, period)
f_kalman_filter(src) =>
    float value1= na
    float value2 = na
    value1 := 0.2 * (src - src[1]) + 0.8 * nz(value1[1])
    value2 := 0.1 * (ta.tr) + 0.8 * nz(value2[1])
    lambda = math.abs(value1 / value2)
    alpha = (-math.pow(lambda, 2) + math.sqrt(math.pow(lambda, 4) + 16 * math.pow(lambda, 2)))/8
    value3 = float(na)
    value3 := alpha * src + (1 - alpha) * nz(value3[1])
//SWITCH
ma_func(modeSwitch, src, len, use_k_f=true) =>
      modeSwitch == "SMA"   ? use_kalman_filter and use_k_f ? f_kalman_filter(ta.sma(src, len))  : ta.sma(src, len) :
      modeSwitch == "RMA"   ? use_kalman_filter and use_k_f ? f_kalman_filter(ta.rma(src, len))  : ta.rma(src, len) :
      modeSwitch == "EMA"   ? use_kalman_filter and use_k_f ? f_kalman_filter(ta.ema(src, len))  : ta.ema(src, len) :
      modeSwitch == "TEMA"  ? use_kalman_filter and use_k_f ? f_kalman_filter(tema(src, len))    : tema(src, len):
      modeSwitch == "DEMA"  ? use_kalman_filter and use_k_f ? f_kalman_filter(f_dema(src, len))  : f_dema(src, len):
      modeSwitch == "ZLEMA" ? use_kalman_filter and use_k_f ? f_kalman_filter(f_zlema(src, len)) : f_zlema(src, len):
      modeSwitch == "WMA"   ? use_kalman_filter and use_k_f ? f_kalman_filter(ta.wma(src, len))  : ta.wma(src, len):
      modeSwitch == "VWMA"  ? use_kalman_filter and use_k_f ? f_kalman_filter(ta.vwma(src, len)) : ta.vwma(src, len):
      modeSwitch == "Hma"   ? use_kalman_filter and use_k_f ? f_kalman_filter(HMA(src, len))     : HMA(src, len):
      modeSwitch == "Ehma"  ? use_kalman_filter and use_k_f ? f_kalman_filter(EHMA(src, len))    : EHMA(src, len):
      modeSwitch == "Thma"  ? use_kalman_filter and use_k_f ? f_kalman_filter(THMA(src, len/2))  : THMA(src, len/2):
      modeSwitch == "ATR"   ? use_kalman_filter and use_k_f ? f_kalman_filter(ATR_func(src, len)): ATR_func(src, len) :
      modeSwitch == "L"   ? use_kalman_filter and use_k_f ? f_kalman_filter(ta.lowest(len)): ta.lowest(len) :
      modeSwitch == "H"   ? use_kalman_filter and use_k_f ? f_kalman_filter(ta.highest(len)): ta.highest(len) :
      modeSwitch == "DMA"   ? donchian(len) : na

//Var
sum = 0.0
maopening = 0.0
maclosing = 0.0
os = maopeningsrc
cs = maclosingsrc
pos = strategy.position_size
p = 0.0
p := pos == 0 ? (strategy.equity / 100) / close : p[1]
truetime = true
loss = 0.0
maxloss = 0.0
equity = 0.0

//MA Opening
maopening := ma_func(maopeningtyp, maopeningsrc, maopeninglen)

//MA Closing
maclosing := ma_func(maclosingtyp, maclosingsrc, maclosinglen) * maclosingmul

long1 = long1on == false ? 0 : long1shift == 0 ? 0 : long1lot == 0 ? 0 : maopening == 0 ? 0 : maopening * long1shift
short1 = short1on == false ? 0 : short1shift == 0 ? 0 : short1lot == 0 ? 0 : maopening == 0 ? 0 : maopening * short1shift
//Colors
maopeningcol = maopening == 0 ? na : color.blue
maclosingcol = maclosing == 0 ? na : color.fuchsia
long1col = long1 == 0 ? na : color.green
short1col = short1 == 0 ? na : color.red
//Lines
plot(maopening, offset = OFFS, color = maopeningcol)
plot(maclosing, offset = OFFS, color = maclosingcol)
long1line = long1 == 0 ? close : long1
short1line = short1 == 0 ? close : short1
plot(long1line, offset = OFFS, color = long1col)
plot(short1line, offset = OFFS, color = short1col)

//Lots
lotlong1 = p * long1lot
lotshort1 = p * short1lot

//Entry
if maopening > 0 and maclosing > 0 and truetime
    //Long
    sum := 0
    strategy.entry("L", strategy.long, lotlong1, limit = on_close ? na : long1, when = long1 > 0 and pos <= sum and (on_close ? close <= long1[trade_offset] : true))
    sum := lotlong1

    //Short
    sum := 0
    pos := -1 * pos
    strategy.entry("S", strategy.short, lotshort1, limit = on_close ? na : short1, when = short1 > 0 and pos <= sum and (on_close ? close >= short1[trade_offset] : true))
    sum := lotshort1

strategy.exit("Exit", na, limit = maclosing)
if time > finalTime
    strategy.close_all()

Plus de