Trois moyennes mobiles croisées et stratégie des indicateurs Williams

Auteur:ChaoZhang est là., Date: 28 septembre 2023 à 10h58
Les étiquettes:

Résumé

Cette stratégie identifie la direction de la tendance des prix en combinant trois moyennes mobiles lisses, l'indice de force relative (RSI) et l'indicateur Williams, et cherche des opportunités de trading lorsque la tendance s'inverse. Elle va long (short) lorsque les moyennes mobiles rapides, moyennes et lentes s'alignent vers le haut (vers le bas), le RSI est supérieur (inférieur) à 50 et un signal Williams vers le bas (vers le haut) apparaît. Le stop loss est fixé à un certain pourcentage du prix d'entrée et le profit est réalisé à un certain pourcentage de mouvement dans la direction favorable du prix d'entrée.

La logique de la stratégie

La stratégie utilise trois moyennes mobiles avec des périodes différentes, y compris des MA rapides, moyennes et lentes. Lorsque le MA rapide traverse le MA moyen, il signale une tendance haussière des prix. Lorsque le MA rapide traverse en dessous du MA moyen, il signale une tendance baissière des prix. Après avoir identifié la tendance haussière ou la tendance baissière, la stratégie attend la première opportunité de trading.

Plus précisément, une fois que le prix entre dans une tendance haussière, la stratégie attend que les cinq conditions suivantes soient remplies avant d'aller long:

  1. Les MAs rapides, moyennes et lentes pointent vers le haut;

  2. Le RSI est supérieur à 50;

  3. Une tendance à la baisse de Williams apparaît;

  4. Le prix franchit la marge de manœuvre lente;

  5. Il n'y a pas de position actuelle.

Une fois que le prix entre dans une tendance à la baisse, la stratégie attend que les cinq conditions suivantes soient remplies avant d'aller court:

  1. Les MAs rapides, moyennes et lentes sont toutes pointées vers le bas;

  2. RSI inférieur à 50;

  3. Un schéma Williams à la hausse apparaît;

  4. Le prix dépasse la marge de manœuvre lente;

  5. Il n'y a pas de position actuelle.

Après avoir fait long ou court, la stratégie fixe un stop loss à un certain pourcentage en dessous du prix d'entrée et un objectif de profit à un certain pourcentage au-dessus du prix d'entrée.

Les avantages

  1. La combinaison de plusieurs indicateurs pour confirmer les entrées peut éviter efficacement les fausses ruptures. Les triple MA identifient la direction de la tendance, Williams capte les signaux d'inversion et le RSI filtre l'action des prix dans la plage, améliorant conjointement la précision des entrées.

  2. La mise en place d'un stop loss et d'une prise de profit permet de contrôler le risque/récompense de chaque transaction, ce qui garantit que les transactions gagnantes dépassent les transactions perdantes.

  3. La logique de la stratégie est claire et facile à comprendre. Les paramètres sont raisonnablement définis. Il convient aux traders à différents niveaux.

Les risques

  1. Les indicateurs peuvent générer des signaux incorrects pendant les marchés à plage, provoquant des entrées inutiles.

  2. Le croisement rapide et moyen de l'AM peut présenter de fausses éruptions.

  3. Si le stop loss est trop proche du prix d'entrée, il peut être arrêté prématurément.

  4. Si le profit est trop éloigné du prix d'entrée, il peut ne pas être atteint.

Directions d'optimisation

  1. Testez différentes combinaisons de paramètres pour les trois MA et RSI.

  2. Ajoutez d'autres indicateurs, comme le volume, pour vérifier si le volume augmente lors des ruptures.

  3. Paramètres d'essai basés sur différents produits.

  4. Tracer des courbes de profit basées sur les résultats des tests antérieurs pour optimiser le stop loss et le profit.

  5. Essayez le trading papier avant de lui permettre d'optimiser les paramètres.

Conclusion

La stratégie a une logique générale claire, entrant et sortant des positions avec une combinaison d'indicateurs, ce qui contrôle efficacement le risque. Il y a beaucoup de place pour l'optimisation des paramètres. En testant différents paramètres, cette stratégie peut devenir une stratégie de trading quantitative rentable. Cependant, aucune stratégie ne peut éviter complètement les pertes. Les traders doivent suivre des disciplines de trading - prendre des profits en gagnant et réduire les pertes en perdant.


/*backtest
start: 2023-08-28 00:00:00
end: 2023-09-27 00:00:00
period: 3h
basePeriod: 15m
exchanges: [{"eid":"Futures_Binance","currency":"BTC_USDT"}]
*/

//This script is a combination of 3 smoothed moving averages, and RSI. When moving averages are aligned upward (downward) and RSI is above (below) 50 and a down (up) William fractal appears, it enters long (short) position. Exiting from long and short entries are defined by StopLoss and TargetProfit.

//@version=5

strategy(title="3SmmaCrossUp + Fractal + RSI", overlay=true, default_qty_type=strategy.percent_of_equity, default_qty_value=100, currency=currency.USD, commission_type=strategy.commission.percent, commission_value=0.03)

///////////////////////////////////////////////////////////////////////////////////////////////////////////////// inputs

// Global
src = input(close, title="Source")
stopLoss = input.float(defval = 0.1, title = "Stop Loss %", minval = 0, maxval=100, step = 0.1)
targetProfit = input.float(defval = 0.4, title = "Target Profit %", minval = 0, maxval=100, step = 0.1)

// Smooth Moving Average
fastSmmaLen = input.int(21, minval=1, title="Fast Length", group = "Smooth Moving Average")
midSmmaLen = input.int(50, minval=1, title="Mid Length",group = "Smooth Moving Average")
slowSmmaLen = input.int(200, minval=1, title="Slow Length",group = "Smooth Moving Average")

// RSI
rsiLen = input.int(defval=14, title="length", minval=1, maxval=1000, step=1, group="RSI")

// Fractals
n = input.int(title="Periods", defval=2, minval=2, group = "Fractals")

///////////////////////////////////////////////////////////////////////////////////////////////////////////////// initialization

var waitingFirstTradeInUpwardTrend = false
var waitingFirstTradeInDownwardTrend = false

///////////////////////////////////////////////////////////////////////////////////////////////////////////////// functions

smma(ma, src, len) => 
    smma = 0.0
    smma := na(smma[1]) ? ma : (smma[1] * (len - 1) + src) / len
    smma
    
fractals(n, highs, lows) =>
    // UpFractal
    bool upflagDownFrontier = true
    bool upflagUpFrontier0 = true
    bool upflagUpFrontier1 = true
    bool upflagUpFrontier2 = true
    bool upflagUpFrontier3 = true
    bool upflagUpFrontier4 = true
    for i = 1 to n
        upflagDownFrontier := upflagDownFrontier and (highs[n-i] < highs[n])
        upflagUpFrontier0 := upflagUpFrontier0 and (highs[n+i] < highs[n])
        upflagUpFrontier1 := upflagUpFrontier1 and (highs[n+1] <= highs[n] and highs[n+i + 1] < highs[n])
        upflagUpFrontier2 := upflagUpFrontier2 and (highs[n+1] <= highs[n] and highs[n+2] <= highs[n] and highs[n+i + 2] < highs[n])
        upflagUpFrontier3 := upflagUpFrontier3 and (highs[n+1] <= highs[n] and highs[n+2] <= highs[n] and highs[n+3] <= highs[n] and highs[n+i + 3] < highs[n])
        upflagUpFrontier4 := upflagUpFrontier4 and (highs[n+1] <= highs[n] and highs[n+2] <= highs[n] and highs[n+3] <= highs[n] and highs[n+4] <= highs[n] and highs[n+i + 4] < highs[n])
    flagUpFrontier = upflagUpFrontier0 or upflagUpFrontier1 or upflagUpFrontier2 or upflagUpFrontier3 or upflagUpFrontier4
    
    upFractal = (upflagDownFrontier and flagUpFrontier)
    
    // downFractal
    bool downflagDownFrontier = true
    bool downflagUpFrontier0 = true
    bool downflagUpFrontier1 = true
    bool downflagUpFrontier2 = true
    bool downflagUpFrontier3 = true
    bool downflagUpFrontier4 = true
    
    for i = 1 to n
        downflagDownFrontier := downflagDownFrontier and (lows[n-i] > lows[n])
        downflagUpFrontier0 := downflagUpFrontier0 and (lows[n+i] > lows[n])
        downflagUpFrontier1 := downflagUpFrontier1 and (lows[n+1] >= lows[n] and lows[n+i + 1] > lows[n])
        downflagUpFrontier2 := downflagUpFrontier2 and (lows[n+1] >= lows[n] and lows[n+2] >= lows[n] and lows[n+i + 2] > lows[n])
        downflagUpFrontier3 := downflagUpFrontier3 and (lows[n+1] >= lows[n] and lows[n+2] >= lows[n] and lows[n+3] >= lows[n] and lows[n+i + 3] > lows[n])
        downflagUpFrontier4 := downflagUpFrontier4 and (lows[n+1] >= lows[n] and lows[n+2] >= lows[n] and lows[n+3] >= lows[n] and lows[n+4] >= lows[n] and lows[n+i + 4] > lows[n])
    flagDownFrontier = downflagUpFrontier0 or downflagUpFrontier1 or downflagUpFrontier2 or downflagUpFrontier3 or downflagUpFrontier4
    
    downFractal = (downflagDownFrontier and flagDownFrontier)
    [upFractal, downFractal]

///////////////////////////////////////////////////////////////////////////////////////////////////////////////// calcs

[upFractal, downFractal] = fractals(n, high, low)


rsiIsHigh = ta.rsi(src, rsiLen) >= 50 


slowMa = ta.sma(src, slowSmmaLen)
midMa = ta.sma(src, midSmmaLen)
fastMa = ta.sma(src, fastSmmaLen)

slowSmma = smma(slowMa ,src, slowSmmaLen)
midSmma = smma(midMa, src, midSmmaLen)
fastSmma = smma(fastMa, src, fastSmmaLen)

isFastSmmaUpward = ta.rising(fastSmma, 1)
isMidSmmaUpward = ta.rising(midSmma, 1)
isSlowSmmaUpward = ta.rising(slowSmma, 1)

isFastSmmaDownward = ta.falling(fastSmma, 1)
isMidSmmaDownward = ta.falling(midSmma, 1)
isSlowSmmaDownward = ta.falling(slowSmma, 1)

slowMovingAveragesAreUpward = isMidSmmaUpward and isSlowSmmaUpward
slowMovingAveragesAreDownward = isMidSmmaDownward and isSlowSmmaDownward

justEnteredUpwardTrend = ta.crossover(fastSmma, midSmma) ? true : false
justEnteredDownwardTrend = ta.crossunder(fastSmma, midSmma) ? true : false

waitingFirstTradeInUpwardTrend := justEnteredUpwardTrend == true ? true : (isFastSmmaDownward or isMidSmmaDownward or isSlowSmmaDownward ? false : waitingFirstTradeInUpwardTrend)
waitingFirstTradeInDownwardTrend := justEnteredDownwardTrend == true ? true : (isFastSmmaUpward or isMidSmmaUpward or isSlowSmmaUpward ? false : waitingFirstTradeInDownwardTrend)

priceCrossedOverSlowMa = ta.crossover(close, slowSmma)
priceCrossedUnderSlowMa = ta.crossunder(close, slowSmma)

enterLongCondition = barstate.isconfirmed and low > fastSmma and rsiIsHigh and (downFractal or priceCrossedOverSlowMa) and waitingFirstTradeInUpwardTrend and strategy.position_size == 0

enterShortCondition = barstate.isconfirmed and high < fastSmma and (not rsiIsHigh) and (upFractal or priceCrossedUnderSlowMa) and waitingFirstTradeInDownwardTrend and strategy.position_size == 0

///////////////////////////////////////////////////////////////////////////////////////////////////////////////// strategy

if(enterLongCondition)
    strategy.entry(id="L", direction=strategy.long)
    waitingFirstTradeInUpwardTrend := false

if(enterShortCondition)
    strategy.entry(id="S", direction=strategy.short)
    waitingFirstTradeInDownwardTrend := false
    
if(strategy.position_size > 0)
    strategy.exit(id="EL", stop=strategy.position_avg_price * (1 - stopLoss/100), limit=strategy.position_avg_price * (1+targetProfit/100)) 
if(strategy.position_size < 0)
    strategy.exit(id="ES", stop=strategy.position_avg_price * (1 + stopLoss/100), limit=strategy.position_avg_price * (1-targetProfit/100)) 

///////////////////////////////////////////////////////////////////////////////////////////////////////////////// plots

plot(series = slowSmma, title="Slow SMMA", linewidth=3)
plot(series = midSmma, title="Mid SMMA", linewidth=2)
plot(series = fastSmma, title="Fast SMMA", linewidth=1)
plotchar(series=rsiIsHigh, title='rsiIsHigh', char='')
plotchar(series=justEnteredUpwardTrend, title='justEnteredUpwardTrend', char='')
plotchar(series=justEnteredDownwardTrend, title='justEnteredDownwardTrend', char='')
plotchar(series=waitingFirstTradeInUpwardTrend, title='waitingFirstTradeInUpwardTrend', char='')
plotchar(series=waitingFirstTradeInDownwardTrend, title='waitingFirstTradeInDownwardTrend', char='')
plotchar(series=enterLongCondition, title='enterLongCondition' , char='')
plotchar(series=enterShortCondition, title='enterShortCondition' , char='')
plotshape(series=upFractal, title='upFractal', style=shape.triangleup, location=location.abovebar, color=#009688, size = size.tiny)
plotshape(series=downFractal, title='downFractal', style=shape.triangledown, location=location.belowbar, color=color.red, size = size.tiny)













Plus de